MyBatis基础之执行SQL

2023-09-20 11:52:16

在这里插入图片描述

执行 SQL 语句

Mapper 是 MyBatis 最强大的工具与功能,它用于执行 SQL 语句。

<?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="...">
  <insert id="..."> ... </insert>
  <delete id="..."> ... </delete>
  <update id="..."> ... </update>
  <select id="..."> ... </select>
</mapper>
元素描述备注
select查询语句,常用又复杂可以自定义参数,返回结果集等
insert插入语句执行后返回一个整数,代表插入的条数
update更新语句执行后返回一个整数,代表更新的条数
delete删除语句执行后返回一个整数,代表删除的条数
resultMap定义查询结果映射关系,常用又复杂它将提供映射规则

1. 增删改操作

insert 元素

insert 元素的必要属性有 :

元素名说明
id和 Mapper 的 namespace 组合起来是唯一的,提供给 MyBatis 调用。
parameterType类的完全限定名,或内置/自定义的类的别名(Alias)。
可以向 SQL 传递 JavaBean 和 Map 等复杂参数类型,但 Map 参数不建议使用

MyBatis 在执行插入之后会返回一个 整数,以表示插入的记录数。

<insert id="insertDepartment" parameterType="com.xja.hemiao.bean.Department">
  INSERT INTO dept(dname, loc) VALUES(#{dname}, #{loc});
</insert>

在传递参数时,MyBatis 中可用的占位符有两种:#{}${}

  • #{}:MyBatis 使用的是 JDBC 中的 PreparedStatement
  • ${}:MyBatis 使用的是 JDBC 中的 Statement
--   #{}是占位符,会进行预编译,传参 'admin' or '1'='1', 0
-- ${}是直接替换,可能会改变你的sql语法(sql注入)传参  
select * from users where username=#{} and password = #{}
                    username = '' admin' or '1'='1'' and password = '0'

select * from users where username=${} and password =${}
                       username = 'admin' or '1'='1' and password = '0'

--  从消耗来说,二者单次的消耗是相同的,批处理时预编译则更友好,缓存重复利用,
-- 当然 就算是#{}也不是完全保证sql的安全,之后可以增加逻辑判断,密码加密等手段
// 这是一个相较于 classpath 的文件路径名。而且,不需要使用 / 。
InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
SqlSessionFactory factory = new SqlSessionFactoryBuilder().build(is);
SqlSession session = factory.openSession();

Department dept = new Department("Test", "BeiJing");
int n = session.insert("xxx.yyy.zzz.insertDepartment", dept);

System.out.println(n);      // 输出打印 1

session.commit();
session.close();

insert 过程中的主键回填

大多数情况下,插入信息的主键是由数据库底层生成的,在插入数据后,我们往往需要这个主键,以便于未来的操作。为此,MyBatis 提供了主键回填功能。这个功能需要数据库和数据库驱动支持,MyBatis 才能正常使用它。

开启主键回填功能的 insert 必要属性:

属性说明
useGeneratedKeys启用主键回填功能的开关属性。"true"
keyProperty指定需要回填的 Bean 属性(对应数据库主键列的那个属性)

如果你有大量的 insert 都要用到主键回填功能,而你又觉得要在所有的这些 <insert ...> 中每一个都去写 userGeneratedKeys="true" 很麻烦,MyBatis 的核心配置文件中的 <settings> 中有一个全局设置,可以帮你批量开启所有的 <insert ...> 的主键回填功能:

<settings>
  ...
  <!-- 默认值是 false 。因此你要一个个地写。 -->
  <setting name="useGeneratedKeys" value="true" />
  ...
</settings>

[!attention] 注意
留意 <settings> 的位置。MyBatis 对配置文件的内容的先后顺序有要求。

delete 元素 和 update 元素

和 insert 元素一样,MyBatis 执行完 update 元素 和 delete 元素后会返回一个整数,标示执行后影响的记录条数。

<update id="updateDepartment" parameterType="dept">
  UPDATE dept SET dname = #{dname}, loc = #{loc} WHERE deptno = #{deptno}
</update>

<delete id="deleteDepartment" parameterType="int">
  DELETE FROM dept WHERE deptno = #{deptno}
</delete>
....
session.insert("xxx.yyy.zzz.insertDepartment", dept);
....
session.update("xxx.yyy.zzz.updateDepartment", dept);
....
session.delete("xxx.yyy.zzz.deleteDepartment", 41);
....

2. getMapper 方法

通过 SqlSession 的 insert、upate、delete 和 selectOne、selectList 方法可以去调用 Mapper.xml 中定义的配置文件,进而操作数据库。不过,MyBatis 提供了更『高端』的操作,『帮』程序员去实现 DAO 层代码。

如果将 Mapper.xml 配置文件的 namespace 故意写的和一个 DAO 接口的完全路径名一样,并且该接口中的方法名有“碰巧”和 Mapp.xml 配置文件中的各个 SQL 语句的 id 值一样,那么 MyBatis 就会去为该接口动态生成一个实现类。

通过 SqlSession 的 getMapper 方法传入接口的类对象,就可以获得这个由 MyBatis 动态生成的 DAO 接口的实现类。

3 个“保持一致”:

  • Mapper.xmlnamespace 与接口的完全限定名保持一致。

  • SQL 语句的 id 属性值与接口中的方法名保持一致。

  • SQL 语句的 parameterType 属性值与接口的方法的参数类型保持一致。

package xxx.yyy.zzz.dao;

public interface DepartmentDao {
    public List<Department> listDepartments();
    ...
}
<mapper namespace="xxx.yyy.zzz.dao.DepartmentDao"> <!-- 注意,此处是接口的完全限定名字。-->
  <select id="listDepartments" resultType="dept">
    SELECT * FROM dept
  </select>
  ...
</mapper>
DepartmentDao dao = session.getMapper(DepartmentDao.class);

List<Department> list = dao.listDepartments();

3. 查操作

select 元素

通过 MyBatis 执行 SQL 后,MyBatis 通过其强大的映射规则,可以自动地将返回的结果集绑定到 JavaBean 中。

select 元素的必要属性有:

属性说明
id和 Mapper 的 namespace 组合起来必须唯一
parameterType类的完全限定名,或内置/自定义的类的别名(Alias)
。可以向 SQL 传递 JavaBean 和 Map 等复杂参数类型。
resultType类的完全限定名,查询结果将通过固定规范进行映射;
或者定义为 int、double、float 等参数。
resultMapresultType 的「高级版」,允许我们自定义映射规则。
不能与 resultType 同时使用

select 与 聚合函数

并不是所有的 select 语句都会返回一行,或多行记录。例如,在 select 中使用聚合函数。这种情况对于 MyBatis 而言最为简单,因为不需要将结果集映射成 JavaBean ,它只需要返回一行一列的单个数据。

<select id="getMaxSal" resultType="int">
  SELECT * FROM dept WHERE deptno = #{deptno}
</select>

<!-- 此处特意是 string 类型,以验证效果 -->
<select id="getEmployeeCount" resultType="string">
  SELECT count(empno) FROM emp;
</select>
int n = session.selectOne("xxx.yyy.zzz.getMaxSal");
System.out.println(n);

String str = session.selectOne("xxx.yyy.zzz.getemployeeCount");
System.out.println(str);

MyBatis 可以很智能地将返回结果转换为你所指定的类型,如:int、String 等。

4. 传递多个参数

多参数的传递有三种方法:

传递方式说明
使用 Map 传参不建议使用(已被 JavaBean 传参替代)
使用 JavaBean 传参大量多参 传递时使用
使用 注解 传参少量多参 传递时使用

使用 Map 传递多参数

MyBatis 支持 Map 对象作为参数,此时,要求 select 元素的 parameterType 值为 map

List<Employee> selectBySal1(Map<String, Integer> salMap);
<select id="selectBySal1" parameterType="map" resultType="Employee">
  select * from emp where sal >= #{minSal} and sal &lt; #{maxSal}
</select>
Map<String, Integer> map = new HashMap<>();
map.put("minSal", 1000);
map.put("maxSal", 2000);

List<Employee> list = dao.selectBySal1(map);
for (Employee emp : list) {
    log.info("{}", emp);
}

使用 JavaBean 传递多参

由于 Map 的无语义性,因此官方 不建议使用 Map 传参!

此时,要求 select 元素的 parameterType 属性值为 JavaBean 的完全限定名(或别名)。

@Data
public class SallaryRegion {
    private Integer minSallary;
    private Integer maxSallary;
}
List<Employee> selectBySal2(SallaryRegion region);
<select id="selectBySal2"
        parameterType="com.microboom.bean.po.SallaryRegion"
        resultType="Employee">
    select * 
    from emp 
    where sal >= #{minSallary} 
      and sal &lt; #{maxSallary}
</select>

使用注解方式传递多参数

如果所有的多参数传递都通过定义并使用 JavaBean 来进行,那么项目中会出现大量的参数 JavaBean 的定义,显然这也并不太合理。

为此,Mybatis 提供了参数注解,以减少参数 JavaBean 的定义。

List<Employee> selectBySal3(
    @Param("xxx") Integer minSallary,
    @Param("yyy") Integer MaxSallary);
<select id="selectBySal3" resultType="Employee">
  select * 
  from emp 
  where sal >= #{xxx} 
    and sal &lt; #{yyy}
</select>

补充MyBatis 框架的注解功能相对而言比较薄弱,官方推荐使用 XML 配置,而非注解,但是少量的多参数传递,是 必须使用注解 的场景。

更多推荐

Spring工厂模式,模拟ApplicationContext实现

一、什么是SpringSpring是一个轻量级的Java企业开发解决方案。它集合了众多优秀的设计模式如:单例、代理、工厂模式等。在这其中最基础的也就是工厂模式。二、Spring为什么是轻量级在Spring问世之前JavaEE开发主流使用EnterpriseJavaBean(EJB)来进行。但是这个框架是重量级的,扩展性

正则表达式的学习笔记

[!note]其实这个正则表达式整体上不难,自从这个gpt出来之后这种正则表达式已经不需要我们去写了,我们并不需要自己能够去写特别深奥的代码,我们可以将这个正则表达式交给gpt去做,我们只需要能够看懂就行了,所以学习这个正则表达式,自己写不出来那种比较难的正则没有事情,只要你能够看懂别人给你的,你就达标了正则表达式的练

redis学习

Redis​redis是一个开源的使用ansic语言编写、支持网络、可基于内存亦可持久化的日志型、key-value数据库,并提供多种语言的api。从2010年3月15日起,redis的开发工作由vmware主持.和MySQL一样都是数据仓库,用于存放数据的仓库,存储和管理磁盘查询数据时顺序IO随机IOmysqI底层使

Leetcode算法入门与数组丨5. 数组二分查找

文章目录1二分查找算法2二分查找细节3二分查找两种思路3.1直接法3.2排除法1二分查找算法二分查找算法是一种常用的查找算法,也被称为折半查找算法。它适用于有序数组的查找,并通过将待查找区间不断缩小一半的方式来快速定位目标值。算法思想如下:首先,确定待查找数组的起始位置(通常为数组的第一个元素)和结束位置(通常为数组的

微服务08-认识和使用SpringAMQP

1.AMQP的认识1.1介绍AMQP是什么?看完你就知道了_hello_读书就是赚钱的博客-CSDN博客_amqp好处:什么connection:消息队列的连接、channel:服务发送接收消息的通道、Queue:消息队列——>这些你都不需要自己编写工作过程:发布者(Publisher)发布消息(Message),经由

常见的查找算法以及分块搜索算法的简明教程

顺序查找最基本的查找算法举例//顺序查找publicstaticintsearchSequence(int[]arr,inttarget){inti=0;for(intarr2:arr){if(arr2==target){returni;}i++;}return-1;}二分查找[!warning]值得注意的是这个二分查

Go的性能优化建议

前言:\textcolor{Green}{前言:}前言:💞这个专栏就专门来记录一下寒假参加的第五期字节跳动训练营💞从这个专栏里面可以迅速获得Go的知识Go的性能优化建议3性能优化建议3.1性能优化建议-Benchmark3.2性能优化建议-slice3.3性能优化建议-Map3.4性能优化建议-字符串处理3.5性能

【AI视野·今日CV 计算机视觉论文速览 第248期】Mon, 18 Sep 2023

AI视野·今日CS.CV计算机视觉论文速览Mon,18Sep2023Totally83papers👉上期速览✈更多精彩请移步主页Interesting:📚Robuste-NeRF,处理高速且大噪声事件相机流的NERF模型。(fromNUS新加坡国立)稀疏噪声事件与稠密事件数据的区别:模型架构:项目网站:https:

Python网络编程(socket)

网络编程指的是:在程序中实现两台计算机之间的通信。Python提供了大量网络编程的工具和库,本文重点学习socket和select模块。网络编程涉及许多关于TCPIP的基础知识,本文默认对这些知识已经了解了,不再对TCPIP相关的知识进行学习。socket模块这个模块提供了访问BSD套接字的接口。在所有现代Unix系统

开始在 Windows 上将 Python 用于 Web 开发

🎬岸边的风:个人主页🔥个人专栏:《VUE》《javaScript》⛺️生活的理想,就是为了理想的生活!目录设置开发环境安装适用于Linux的Windows子系统设置VisualStudioCode创建新项目安装Python、pip和venv创建虚拟环境打开WSL-Remote窗口安装MicrosoftPython扩

【C# 基础精讲】抽象类与接口

抽象类(AbstractClass)和接口(Interface)是面向对象编程中两种重要的概念,它们用于定义类的结构、行为和关系,是实现多态性、代码复用和系统设计的关键手段。在C#及其他面向对象编程语言中,抽象类和接口都发挥着重要作用。本文将详细解释抽象类和接口的概念、特点、用法以及在C#中的应用。1.抽象类的概念与特

热文推荐