从头开始学MyBatis—02基于xml和注解分别实现的增删改查

news2025/1/10 12:04:34

首先介绍此次使用的数据库结构,然后引出注意事项。

通过基于xml和基于注解的方式分别实现了增删改查,还有获取参数值、返回值的不同类型对比,帮助大家一次性掌握两种代码编写能力。

目录

数据库

数据库表

实体类

对应的实体类如下:

注意事项

1)两种占位符的说明

①#{}

②${}

2)增、删、改的返回值说明

3)查询操作

①必须指定resultType或是resultMap

4)对应关系

①数据库名与类名不一致的情况

②字段名与属性名不一致的情况

1.获取参数值的方式

1.1基于xml

1.1.1单个参数

①mapper

②xml

③test

④结果

1.1.2多个参数

①mapper

②xml

写法一

写法二

③test

④结果

1.1.3map参数

①mapper

②xml

③test

④结果

1.1.4实体类参数

①mapper

②xml

③test

④结果

1.1.5使用@Param标识参数

①mapper

写法一

写法二

错误写法

②xml

③test

④结果

1.2基于注解

1.2.1单个参数

①mapper

写法一

写法二

②test

③结果

1.2.2多个参数

①mapper

 ②test

③结果

1.2.3map参数

①mapper

②test

③结果

1.2.4实体类参数

①mapper

②test

③结果

1.2.5使用@Param标识参数

①mapper

写法一

写法二

②test

③结果

2.各种操作功能

2.1基于xml

2.1.1增

①mapper

②mapper.xml

③test

④结果

主键id自增并返回值

mapper

mapper.xml

test

结果

2.1.2删

①mapper

②mapper.xml

③test

④结果

删除前

删除后

2.1.3改

①mapper

②mapper.xml

③test

④结果

修改前

修改后

2.1.4查

①mapper

②mapper.xml

③test

④结果

2.2基于注解

2.2.1增

①mapper

②test

③结果

主键id自增并返回结果

mapper

test

结果

2.2.2删

①mapper

②test

③结果

删除前

删除后

2.2.3改

①mapper

②test

③结果

修改前

修改后

2.2.4查

①mapper

②test

③结果

3.各种返回值类型

3.1基于xml

①查询单条数据

②查询多条数据

3.2基于注解

①查询单条数据

②查询多条数据


数据库

数据库表

此次实验的数据库表结构如下,包含主键id和三个属性

实体类

对应的实体类如下:

import com.baomidou.mybatisplus.annotation.TableName;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

/**
 * ClassName: Book
 * Package: com.ykx.domain
 * Description: mybatis学习的数据库表tbl_book对应的实体类
 */
@Data
@AllArgsConstructor
@NoArgsConstructor
@TableName("tbl_book")
public class Book {
    private Integer id;
    private String type;
    private String name;
    private String description;
}

注意事项

1)两种占位符的说明

①#{}

传入的参数当成一个字符串,会给传入的参数加单引号

能够很大程度上防止sql注入

一般用于替换某个值

②${}

将传入的参数值直接显示生成在sql中,不会自动加引号

预编译之前就已经被变量替换,无法防止sql注入

一般用于替换表、字段名

2)增、删、改的返回值说明

返回值固定位Integer,表示受影响的行数

3)查询操作

①必须指定resultType或是resultMap

不配置别名的时候,值为类的全类名

返回值是集合的时候,只需指定集合的泛型即可,如

List<Book> getAll();

<select id="getAll" resultType="com.ykx.pojo.Book">
    select * from tbl_book
</select>

4)对应关系

①数据库名与类名不一致的情况

②字段名与属性名不一致的情况

1.获取参数值的方式

MyBatis 提供了多种传递参数的方式到 SQL 。

理解参数传递的机制有助于提高代码的可读性、灵活性。

1.1基于xml

1.1.1单个参数

①mapper
Book getById(Integer id);
②xml

在单个字面量参数的情况下,{}内的名称可以任意取

如下面的代码可以把#{id} 改成 #{ID}亦或是其他

<select id="getById" resultType="com.ykx.domain.Book">
    select * from tbl_book where id = #{id}
</select>
③test
@Test
public void testSelect(){
    Book book = bookMapperXML.getById(1);
    System.out.println(book);
}
④结果

1.1.2多个参数

①mapper
Book check(String type,String name);
②xml
写法一
<select id="check" resultType="com.ykx.domain.Book">
    select * from tbl_book where type = #{type} and name = #{name}
</select>
写法二
<select id="check" resultType="com.ykx.domain.Book">
    select * from tbl_book where type = #{param1} and name = #{param2}
</select>
③test
@Test
public void testCheck(){
    Book book = bookMapperXML.check("java","mybatis数据");
    System.out.println(book);
}
④结果

1.1.3map参数

①mapper
Book checkByMap(Map<String,String> map);
②xml
<select id="checkByMap" resultType="com.ykx.domain.Book">
    select * from tbl_book where type = #{type} and name = #{name}
</select>
③test
@Test
public void testCheckByMap(){
    Map<String,String> map = new HashMap<>();
    map.put("type","java");
    map.put("name", "mybatis数据");
    Book book = bookMapperXML.checkByMap(map);
    System.out.println(book);
}
④结果

1.1.4实体类参数

①mapper
Integer insert(Book book);
②xml
<insert id="insert">
    insert into tbl_book (type,name,description) values(#{type},#{name},#{description})
</insert>
③test
@Test
public void testInsert(){
    Book book = new Book(null,"计算机视觉","三维重建","基于深度学习的三维重建");
    Integer ans = bookMapperXML.insert(book);
    System.out.println("受影响的行数:" + ans);
}
④结果

1.1.5使用@Param标识参数

 在字段名和属性名对不上或是参数名和字段名对不上的时候很有用

①mapper
写法一
Book check(@Param("type") String type, @Param("name") String name);
写法二
Book check(@Param("type") String type2, @Param("name") String name2);
错误写法
Book check(String type2, String name2);
②xml
<select id="check" resultType="com.ykx.domain.Book">
    select * from tbl_book where type = #{type} and name = #{name}
</select>
③test
@Test
public void testCheck(){
    Book book = bookMapperXML.check("java","mybatis数据");
    System.out.println(book);
}
④结果

1.2基于注解

1.2.1单个参数

①mapper

单个参数的情况下,接口的参数名和注解里的sql语句参数可以不一致

写法一
@Select("select * from tbl_book where id = #{id}")
Book getById(Integer id);
写法二
@Select("select * from tbl_book where id = #{aa}")
Book getById(Integer id);
②test
@Test
public void testSelect(){
    Book book = bookMapper.getById(78);
    System.out.println(book);
}
③结果

1.2.2多个参数

①mapper

注:在未使用@Param的情况下,参数名不对应或是顺序不一样会导致查询失败

@Select("select * from tbl_book where type = #{type} and name = #{name}")
Book getOne(String type,String name);
 ②test
@Test
public void testGetOne(){
    Book book = bookMapper.getOne("计算机视觉","三维重建");
    System.out.println(book);
}
③结果

1.2.3map参数

①mapper
@Select("select * from tbl_book where type = #{type} and name = #{name}")
Book getByMap(Map<String, String> map);
②test

注:map里的key值要和#{}里的值对应,否则出现查询失败的情况

@Test
public void testByMap(){
    Map<String,String> map = new HashMap<>();
    map.put("type","java");
    map.put("name", "mybatis数据");
    Book book = bookMapper.getByMap(map);
    System.out.println(book);
}
③结果

1.2.4实体类参数

①mapper
@Insert("insert into tbl_book (id,type,name,description) " +
        "values(#{id},#{type},#{name},#{description})")
Integer insert(Book book);
②test
@Test
public void testInsert(){
    Book book = new Book(null,"新增数据","测试新增","不带id的新增测试");
    Integer ans = bookMapper.insert(book);
    System.out.println("受影响的行数:" + ans);
}
③结果

1.2.5使用@Param标识参数

只需要#{}里的值和@Param()里的值一样就行

①mapper
写法一
@Select("select * from tbl_book where type = #{type} and name = #{name}")
Book getOne(@Param("type") String type,@Param("name") String name);
写法二
@Select("select * from tbl_book where type = #{type} and name = #{name}")
Book getOne(@Param("type") String typeaa,@Param("name") String nameaa);
②test
@Test
public void testGetOne(){
    Book book = bookMapper.getOne("计算机视觉","三维重建");
    System.out.println(book);
}
③结果

2.各种操作功能

2.1基于xml

2.1.1增

①mapper
//增:实体对象新增数据
Integer insert(Book book);
②mapper.xml
<insert id="insert">
    insert into tbl_book (id,type,name,description) values(#{id},#{type},#{name},#{description})
</insert>

:这里设置了id主键自增的话,可以不设置和传入id

③test
//增:实体对象新增数据
@Test
public void testInsert(){
    Book book = new Book(71,"计算机视觉","三维重建","基于深度学习的三维重建");
    Integer ans = bookMapperXML.insert(book);
    System.out.println("受影响的行数:" + ans);
}
④结果

主键id自增并返回值

关键点:在xml里需要带上useGeneratedKeys和keyProperty

mapper
Integer autoId(Book book);
mapper.xml
<insert id="autoId" useGeneratedKeys="true" keyProperty="id">
    insert into tbl_book (type,name,description) values(#{type},#{name},#{description})
</insert>
test
//增:主键自增且返回值
@Test
public void testAutoId(){
    Book book = new Book(null,"测试id自增","返回主键id","是否成功获取id的值");
    Integer ans = bookMapperXML.autoId(book);
    System.out.println("受影响的行数:" + ans);
    System.out.println("获取主键自增的id值:" + book.getId());
}
结果

2.1.2删

①mapper
//删:根据id删除数据
Integer delete(Integer id);
②mapper.xml
<delete id="delete">
    delete from tbl_book where id = #{id};
</delete>
③test
//删:根据id删除数据
@Test
public void testDelete(){
    Integer ans = bookMapperXML.delete(68);
    System.out.println("受影响的行数:" + ans);
}
④结果

删除前

删除后

2.1.3改

①mapper
//改:根据id修改数据
Integer update(Book book);
②mapper.xml
<update id="update">
    update tbl_book set type = #{type},name = #{name},description = #{description} where id = #{id};
</update>
③test
//改:根据id修改数据
@Test
public void testUpdate(){
    Book book = new Book(58,"测试修改","mybatis修改","基于xml的修改操作");
    Integer ans = bookMapperXML.update(book);
    System.out.println("受影响的行数:" + ans);
}
④结果

修改前

修改后

2.1.4查

①mapper
@Mapper
public interface BookMapperXML {
    
    //查:根据id查询数据
    Book getById(Integer id);

    //查:查询所有数据
    List<Book> getAll();
    
}
②mapper.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
        PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
        "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.ykx.mapper.BookMapperXML">
    <select id="getById" resultType="com.ykx.domain.Book">
        select * from tbl_book where id = #{id}
    </select>
    
    <select id="getAll" resultType="com.ykx.domain.Book">
        select * from tbl_book
    </select>
</mapper>
③test
@SpringBootTest
public class XMLMybatisTest {

    @Autowired
    private BookMapperXML bookMapperXML;

    //查:根据id查询数据
    @Test
    public void testSelect(){
           Book book = bookMapperXML.getById(1);
    }
    
    //查:查询所有数据
    @Test
    public void testGetAll(){
        List<Book> books = bookMapperXML.getAll();
        for(Book book : books){
            System.out.println(book);
        }
    }

}
④结果

2.2基于注解

2.2.1增

①mapper
@Insert("insert into tbl_book (id,type,name,description) " +
        "values(#{id},#{type},#{name},#{description})")
Integer insert(Book book);
②test
//增:实体对象新增数据
@Test
public void testInsert(){
    Book book = new Book(11,"新增数据","测试新增","带id的新增测试");
    Integer ans = bookMapper.insert(book);
    System.out.println("受影响的行数:" + ans);
}
③结果

主键id自增并返回结果

关键点:在mapper对应的方法上面加上@Options注解,并设置useGeneratedKeys和keyProperty的属性值。这个只能搭配insert语句使用!

mapper
@Options(useGeneratedKeys = true,keyProperty = "id")
@Insert("insert into tbl_book (type,name,description) " +
        "values(#{type},#{name},#{description})")
Integer autoId(Book book);
test
//增:主键自增且返回值
@Test
public void testAutoId(){
    Book book = new Book(null,"新增数据2","测试新增2","不带id的新增测试");
    Integer ans = bookMapper.autoId(book);
    System.out.println("受影响的行数:" + ans);
    System.out.println("获取主键自增的id值:" + book.getId());
}
结果

2.2.2删

①mapper
//删:根据id删除数据
@Delete("delete from tbl_book where id = #{id}")
Integer delete(Integer id);
②test
//删:根据id删除数据
@Test
public void testDelete(){
    Integer ans = bookMapper.delete(8);
    System.out.println("受影响的行数:" + ans);
}
③结果

删除前

删除后

2.2.3改

①mapper
//改:根据id修改数据
@Update("update tbl_book set type = #{type}, name = #{name}, description = #{description} where id = #{id}")
Integer update(Book book);
②test
//改:根据id修改数据
@Test
public void testUpdate(){
    Book book = new Book(62,"测试修改","mybatis修改","基于注解的修改操作");
    Integer ans = bookMapper.update(book);
    System.out.println("受影响的行数:" + ans);
}
③结果
修改前

修改后

2.2.4查

①mapper
//查:根据id查询数据
@Select("select * from tbl_book where id = #{id}")
Book getById(Integer id);

//查:查询所有数据
@Select("select * from tbl_book")
List<Book> getAll();
②test
//查:根据id查询数据
@Test
public void testSelect(){
    Book book = bookMapper.getById(71);
    System.out.println(book);
}

//查:查询所有数据
@Test
public void testGetAll(){
    List<Book> books = bookMapper.getAll();
    for(Book book : books){
        System.out.println(book);
    }
}
③结果

3.各种返回值类型

对于增、删、改操作一般返回值类型为integer,表示数据表受影响的行数。对于查询操作,一般返回值类型为实体类或是集合类型。

前面已经对增、删、改返回integer进行了测试,这里就不再赘述,下面对查询操作的返回值进行一个总结。

3.1基于xml

①查询单条数据

mapper层接口用对应的实体类接收

//查:根据id查询数据
Book getById(Integer id);

xml文件设置resultType为具体的实体类类型

<select id="getById" resultType="com.ykx.domain.Book">
    select * from tbl_book where id = #{id}
</select>

②查询多条数据

mapper层接口用List集合接收,其泛型为对应的实体类类型

//查:查询所有数据
List<Book> getAll();

xml文件设置resultType为具体的实体类类型

<select id="getAll" resultType="com.ykx.domain.Book">
    select * from tbl_book
</select>

3.2基于注解

①查询单条数据

mapper层接口的返回值类型设置为对应的实体类类型即可

//查:根据id查询数据
@Select("select * from tbl_book where id = #{id}")
Book getById(Integer id);

②查询多条数据

mapper层接口的返回值类型设置为List集合,其泛型为对应的实体类类型即可

//查:查询所有数据
@Select("select * from tbl_book")
List<Book> getAll();

原创内容 未经同意禁止转载 如有引用请标明出处

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2136629.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【目标检测】labelimg图像标注软件的使用流程

一、labelimg检测图片标注 1、下载labelimg.exe 链接&#xff1a;https://pan.baidu.com/s/1yk8ff56Xu40-ZLBghEQ5nw 提取码&#xff1a;vj8f 下载的文件是编译好的&#xff0c;可执行的labelImg.exe文件。直接将文件放在windows环境下&#xff0c;双击可执行。&#xff08;如果…

典型BUCK电路学习和设计

手把手教你设计12V3Abuck降压电路-2-相关输入参数讲解_哔哩哔哩_bilibili 这里是输入电容&#xff0c;先过大电容&#xff08;电解电容&#xff09;再过小电容&#xff08;陶瓷贴片电容&#xff0c;高频率波&#xff09; 输出也可以同理 开关电源不能带负载的原因&#xff0c…

uniapp vue3 梯形选项卡组件

实现的效果图&#xff1a; 切换选项卡显示不同的内容&#xff0c;把这个选项卡做成了一个组件&#xff0c;需要的自取。 // 组件名为 trapezoidalTab <template> <view class"pd24"><view class"nav"><!-- 左侧 --><view cla…

--链表--

一.链表的概述 二.逻辑图 三.代码详解 //1.定义关于链表的结构体 #include <iostream> #include <stdlib.h> #include <assert.h> using namespace std; typedef int SLTDateType;//适用于不同的数据类型 typedef struct SListNode {SLTDateType data;//数据…

【Day14-单例设计模式动态代理】

单例设计模式 什么是设计模式&#xff08;Design pattern&#xff09; ? 一个问题通常有n种解法&#xff0c;其中肯定有一种解法是最优的&#xff0c;这个最优的解法被人总结出来了&#xff0c;称之为设计模式。设计模式有20多种&#xff0c;对应20多种软件开发中会遇到的问题…

记录开发一个英语听力训练网站

背景 在当前全球经济衰退的背景下&#xff0c;IT相关的工作在国内的竞争也是越来越激烈&#xff0c;为了能够获得更多的可能性&#xff0c;英语的学习也许能为程序员打开一扇新的窗户&#xff0c;比如很多远程的工作尤其是国际化背景的工作团队&#xff0c;英语的协作沟通是必…

Modbus通信

Modbus是一种经典的工业通信协议&#xff0c;由Modicon&#xff08;现为施耐德电气&#xff09;在1979年首次发布。它广泛应用于各种工业自动化系统中&#xff0c;尤其是在PLC&#xff08;可编程逻辑控制器&#xff09;与其他设备之间的通信。Modbus的主要特点是其简单性和开放…

《Oracle(一)- 基础》

文章目录 一、Oracle简介&#xff08;一&#xff09;什么是ORACLE&#xff08;二&#xff09;ORACLE 体系结构1.数据库2.实例3.数据文件&#xff08;dbf&#xff09;4.表空间5.用户 二、ORACLE 安装与配置&#xff08;一&#xff09;VMware 挂载 windows server 2003&#xff0…

海外短剧系统一站式开发+h5,app双端

前言&#xff1a; 海外短剧是指那些制作时间短、剧情紧凑、内容丰富的剧集&#xff0c;主要在海外市场&#xff08;如北美、欧洲、东南亚等&#xff09;播放并受到欢迎。 而海外短剧系统是指一种用于制作和播放海外短剧的系统。该系统通常由电视台、制片公司或在线视频平台使…

C++:STL详解(一)string类的基本介绍与使用方式

✨ Blog’s 主页: 白乐天_ξ( ✿&#xff1e;◡❛) &#x1f308; 个人Motto&#xff1a;实践是检验真理的唯一标准&#xff01;&#xff01;&#xff01;敲代码需要勤快点&#xff01;&#xff01;&#xff01;&#xff01; &#x1f4ab; 欢迎来到我的学习笔记&#xff0…

十二,Spring Boot 异常处理(自定义异常页面,全局异常,自定义异常)

十二&#xff0c;Spring Boot 异常处理(自定义异常页面&#xff0c;全局异常&#xff0c;自定义异常) 文章目录 十二&#xff0c;Spring Boot 异常处理(自定义异常页面&#xff0c;全局异常&#xff0c;自定义异常)1. 基本介绍2. 自定义异常页面3. 全局异常4. 自定义异常5. 补充…

LineageOS刷机教程

版权归作者所有&#xff0c;如有转发&#xff0c;请注明文章出处&#xff1a;https://cyrus-studio.github.io/blog/ LineageOS 是一个基于 Android 开源项目&#xff08;AOSP&#xff09;的开源操作系统&#xff0c;主要由社区开发者维护。它起源于 CyanogenMod 项目&#xff…

数据库索引底层数据结构之B+树MySQL中的页索引分类【纯理论知识,干货分享,面试必备】

目录 1、索引简介 1.1 什么是索引 1.2 使用索引的原因 2、索引中数据结构的设计 —— B树 2.1 哈希 2.2 二叉搜索树 2.3 B树 2.4 最终选择之——B树 2.4.1 B树与B树的对比(面向索引)【面试题】 3、MySQL中的页 3.1 页的使用原因 3.2 页的结构 3.2.1 页文件头和页文件…

解锁定位服务:Flutter应用中的高德地图定位

前言 在现代移动应用开发中&#xff0c;定位服务已成为一项基本功能&#xff0c;它使得应用能够获取用户的地理位置信息&#xff0c;为用户提供更加个性化的服务。 Flutter 作为跨平台的移动应用开发框架&#xff0c;支持集成多种服务&#xff0c;包括定位服务。 本文将介绍如…

HR8870:可PWM控制,4.5A直流有刷电机驱动数据手册

HR8870芯片描述 HR8870是一款直流有刷电机驱动器&#xff0c;适用于打印机、电器、工业设备以及其他小型机器。两个逻辑输入控制H桥驱动器&#xff0c;该驱动器由四个N-MOS组成&#xff0c;能够以高达4.5A的峰值电流双向控制电机。利用电流衰减模式&#xff0c;可通过对输入进行…

故障码格式解析

中&#xff0c;诊断故障码&#xff08;DTC, Diagnostic Trouble Code&#xff09;是由一个字母前缀和三个后续字符组成的。这些字母前缀根据故障所属的系统类别来区分&#xff0c;具体如下&#xff1a; B0 -- B3&#xff1a;表示车身系统&#xff08;Body&#xff09;的故障码…

Linux CTF逆向入门

1.ELF格式 我们先来看看 ELF 文件头&#xff0c;如果想详细了解&#xff0c;可以查看ELF的man page文档。 关于ELF更详细的说明&#xff1a; e_shoff&#xff1a;节头表的文件偏移量&#xff08;字节&#xff09;。如果文件没有节头表&#xff0c;则此成员值为零。 sh_offset&…

Qt 菜单、工具栏 的基本使用

效果 代码 #include "mainwindow.h" #include "ui_mainwindow.h" #include<QToolBar> #include<QDebug> #include<QPushButton>MainWindow::MainWindow(QWidget *parent): QMainWindow(parent), ui(new Ui::MainWindow) {ui->setupU…

【JAVA入门】Day45 - 压缩流 / 解压缩流

【JAVA入门】Day45 - 压缩流 / 解压缩流 文章目录 【JAVA入门】Day45 - 压缩流 / 解压缩流一、解压缩流二、压缩流 在文件传输过程中&#xff0c;文件体积比较大&#xff0c;传输较慢&#xff0c;因此我们发明了一种方法&#xff0c;把文件里的数据压缩到一种压缩文件中&#x…

【LLMs对抗性提示:提示泄漏、非法行为、DAN、Waluigi效应、 游戏模拟器、防御策略————】

对抗性提示 目录 对抗性提示 提示注入 提示泄漏 非法行为 DAN Waluigi效应 GPT-4模拟器 游戏模拟器 防御策略 在指令中添加防御 参数化提示组件 引用和其他格式 对抗提示检测器 模型类型 参考文献 Adversarial prompting是提示工程中的一个重要主题&#xff0c…