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

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

通过基于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();

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

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Java项目: 基于SpringBoot+mybatis+maven洗衣店订单管理系统(含源码+数据库+开题报告+任务书+毕业论文)

一、项目简介 本项目是一套基于SpringBootmybatismaven洗衣店订单管理系统 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目附带全部源码可作为毕设使用。 项目都经过严格调试&#xff0c;eclipse或者idea 确保可以运行&#xff01; 该系统功能完善、界面美观、操作…

List<Map<String, Object>>汇总统计排序

开发环境&#xff1a;jdk 1.8 需求一&#xff1a; 1、统计每个小时(升序)不同事件的产品产量 2、统计不同事件&#xff08;OK 、NG&#xff09;的总产量 public static void main(String[] args) {//数据源List<Map<String, Object>> list new ArrayList<Map…

根据 IP 地址进行 VPN 分流(详细,亲测,通用)

根据 IP 地址进行 VPN 分流&#xff08;详细&#xff0c;亲测&#xff0c;通用&#xff09; 背景 不在学校的时候需要使用实验室的服务器&#xff0c;但是实验室的服务器只能在校园网内访问&#xff0c;因此在校外就需要使用学校的 VPN&#xff0c;但是打开 VPN 以后会默认将…

js 3个事件监听器 EventListeners

起因&#xff0c; 目的: 我有2个显示器。 某视频网站&#xff0c;我想一边播放视频&#xff0c;一边搞其他。但是&#xff0c;当我把鼠标移动到浏览器外面&#xff0c;点击一下别处&#xff0c; 视频就会自动暂停. 这个叫做 事件监听&#xff01; blur, 在元素或窗口失去焦点…

JSON对接发送短信验证码怎么获取状态报告

现在很多网站的用户注册都会加一个短信验证功能&#xff0c;也就是需要用户填写手机号&#xff0c;然后点击“获取短信验证码”&#xff0c;将收到的短信验证码输入验证通过后方能进行下一步完成注册&#xff0c;现在短信验证码被广泛应用于网站用户注册&#xff0c;还被广泛应…

linux 安装histomicstk

一直安装失败&#xff0c;源码编译也未成功 最后使用这个成功了 pip install histomicstk --find-links https://girder.github.io/large_image_wheels

零基础如何学会Appium自动化测试?

前言 appium是一款移动自动化测试工具&#xff0c;经常被用于实现UI自动化测试&#xff0c;其可支持安卓和IOS两大平台&#xff0c;还支持多种编程&#xff0c;因而得到了广泛的应用。此处便是立足于安卓平台&#xff0c;借助appium工具&#xff0c;使用python语言实现简单的自…

王者荣耀改重复名(java源码)

王者荣耀改重复名 项目简介 “王者荣耀改重复名”是一个基于 Spring Boot 的应用程序&#xff0c;用于生成王者荣耀游戏中的唯一名称。通过简单的接口和前端页面&#xff0c;用户可以输入旧名称并获得一个新的、不重复的名称。 功能特点 生成新名称&#xff1a;提供一个接口…

[mysql]mysql排序和分页

#排序和分页本身是两块内容,因为都比较简单,我们就把它分到通一个内容里. #1排序: SELECT * FROM employees #我们会发现,我们没有做排序操作,但是最后出来的107条结果还是会按顺序发出,而且是每次都一样.这我们就有一个疑惑了,现在我们的数据库是根据什么来排序的,在我们没有进…

【机器学习】--- 自然语言推理(NLI)

引言 随着自然语言处理&#xff08;NLP&#xff09;的迅速发展&#xff0c;**自然语言推理&#xff08;Natural Language Inference, NLI&#xff09;**已成为一项重要的研究任务。它的目标是判断两个文本片段之间的逻辑关系。这一任务广泛应用于机器阅读理解、问答系统、对话…

二叉搜索树(Java实现)

博主主页: 码农派大星. 数据结构专栏:Java数据结构 数据库专栏:MySQL数据库 JavaEE专栏:JavaEE 关注博主带你了解更多数据结构知识 目录 1.概念 2.实现二叉搜索树 定义节点 查找元素 插入元素 删除元素 1.概念 二叉搜索树又称二叉排序树,或者它是一棵空树,或者是具有…

数字IC设计\FPGA 职位经典笔试面试整理--语法篇 Verilog System Verilog(部分)

注&#xff1a; 资料都是基于网上一些博客分享和自己学习整理而成的 Verilog 1. 数据类型 Verilog一共有19种数据类型 基础四种数据类型&#xff1a;reg型&#xff0c;wire型&#xff0c;integer型&#xff0c;parameter型 reg型   reg类型是寄存器数据类型的关键字。寄存…

镀金引线---

一、沉金和镀金 沉金和镀金都是常见的PCB金手指处理方式&#xff0c;它们各有优劣势&#xff0c;选择哪种方式取决于具体的应用需求和预算。 沉金&#xff08;ENIG&#xff09;是一种常用的金手指处理方式&#xff0c;它通过在金手指表面沉积一层金层来提高接触性能和耐腐蚀性…

【C++】模拟实现vector

在上篇中我们已经了解过的vector各种接口的功能使用&#xff0c;接下来我们就试着模拟实现一下吧&#xff01; 注意&#xff1a;我们在此实现的和C标准库中实现的有所不同&#xff0c;其目的主要是帮助大家大概理解底层原理。 我们模拟vector容器的大致框架是&#xff1a; t…

[SIGGRAPH-24] CharacterGen

[pdf | code | proj] LRM能否用于3D数字人重建&#xff1f;问题在于&#xff1a;1&#xff09;缺少3D数字人数据&#xff1b;2&#xff09;重建任意姿态的3D数字人不利于后续绑定和驱动。构建3D数字人数据集&#xff1a;在VRoidHub上采集数据&#xff0c;得到13746个风格化角色…

图片编辑软件,这4款免费又好用!

在这个视觉为王的时代&#xff0c;一张精心编辑的图片往往能瞬间吸引眼球&#xff0c;无论是社交媒体分享、博客配图还是商业宣传&#xff0c;都离不开强大的图片编辑工具。但高昂的软件费用常常让人望而却步。别担心&#xff0c;今天我们就来揭秘4款不仅免费还超级好用的图片编…

OpenAI 刚刚推出 o1 大模型!!突破LLM极限

北京时间 9 月 13 日午夜&#xff0c;OpenAI 正式发布了一系列全新的 AI 大模型&#xff0c;专门用于应对复杂问题。 这一新模型的出现代表了一个重要突破&#xff0c;其具备的复杂推理能力远远超过了以往用于科学、代码和数学等领域的通用模型&#xff0c;能够解决比之前更难的…

在线IP代理检测:保护您的网络安全

在互联网飞速发展的今天&#xff0c;越来越多的人开始意识到网络安全和隐私保护的重要性。在线IP代理检测工具作为一种有效的网络安全手段&#xff0c;能够帮助用户识别和检测IP代理的使用情况&#xff0c;从而更好地保护个人隐私和数据安全。本文将详细介绍在线IP代理检测的相…

​‌Macbook如何玩《黑神话:悟空》‌2024最新详细方法

‌Mac用户可以通过几种方法玩《黑神话&#xff1a;悟空》‌。 ‌使用虚拟机‌&#xff1a;通过Parallels Desktop等虚拟机软件&#xff0c;在Mac上运行Windows系统&#xff0c;并在其中安装和运行《黑神话悟空》。这种方法需要Mac电脑满足游戏的基础配置要求。 不过如果电脑有虚…

AI逻辑推理入门

参考数据鲸 (linklearner.com) 1. 跑通baseline 报名 申领大模型API 模型服务灵积-API-KEY管理 (aliyun.com) 跑通代码 在anaconda新建名为“LLM”的环境,并安装好相应包后,在jupyter notebook上运行baseline01.ipynb 2. 赛题解读 一般情况下,拿到一个赛题之后,我们需…