Java中操作Mybatis编写动态SQL,设置一二级缓存
Mybatis调用流程
MyBatis 是一个优秀的持久层框架,用于处理数据库操作。MyBatis 通过将 SQL 语句与 Java 对象进行映射,使得开发人员能够通过对象来操作数据库,而无需编写底层的 SQL 语句。MyBatis 的工作原理可以概括为以下几个步骤:
- 配置文件加载:MyBatis 读取配置文件,该文件包含了数据库连接信息、Mapper 接口及其映射关系等信息。
- Mapper 接口生成:MyBatis 根据 Mapper 接口生成的 Java 文件,用于封装数据库操作方法。
- SqlSessionFactory 创建:通过配置文件加载 SqlSessionFactory,它是 MyBatis 的核心类,用于创建 SqlSession。
- SqlSession 创建:通过 SqlSessionFactory 创建 SqlSession,它是 MyBatis 的运行时环境,用于执行 SQL 语句。
- Mapper 代理生成:MyBatis 根据 Mapper 接口生成的代理类,用于简化对 Mapper 接口的调用。
- SQL 映射:MyBatis 将 SQL 语句与 Java 对象进行映射,使得开发人员能够通过对象来操作数据库。
- SQL 执行:MyBatis 通过 SqlSession 执行 SQL 语句,并将结果映射到 Java 对象中。
- 数据持久化:MyBatis 将 Java 对象持久化为数据库中的数据,实现了数据的增删改查操作。
总之,MyBatis 通过将 SQL 语句与 Java 对象进行映射,简化了对数据库的操作,提高了开发效率。
1 Mybatis动态SQL
1 IF 标签语句
if标签会执行所有的if标签,并进行条件判断,满足的部分进行sql拼接
<select id="queryUser" parameterType="map" resultType="user">
select * from usr where 1=1
<if test="id != null" >
and id =#{id}
</if>
<if test="username != null" >
and username = #{username}
</if>
</select>
2 where标签语句
动态拼接where(并且会智能的去掉第一个语句的and或者or)
<!-- where 元素只会在至少有一个子元素的条件返回sql子句的情况下,才去插入"where" 子句-->
<select id="queryUser" parameterType="map" resultType="user">
select * from usr
<where>
<if test="id != null" >
id =#{id}
</if>
<if test="username != null" >
and username = #{username}
</if>
</where>
</select>
3 choose、when、otherwise 标签语句
不会执行所有的when语句,遇到满足的条件,后面的when语句不会执行,otherwise可以不写
<!--
有时不想应用到所有的条件语句,而只想从中择其一项,针对这种情况,Mybatis提供了choose元素
它有点像java中的switch语句
-->
<select id="queryUser" parameterType="map" resultType="user">
select * from usr
<where>
<choose>
<when test="id != null" >
id =#{id}
</when>
<when test="username != null" >
and username = #{username}
</when>
<otherwise>
and id= 4
</otherwise>
</choose>
</where>
</select>
4 set 标签语句
用于更新操作(并且会智能的去掉最后一个语句后面的逗号)
<!--
这里set元素会动态前置set关键字,同时也会删除掉无关的逗号
因为用了条件语句之后很可能就会生成的sql后面留下这些逗号
-->
<update id="updatePwd">
update public."user"
<set>
<if test="name !=null">name=#{name},</if>
<if test="pwd !=null">pwd=#{pwd},</if>
</set>
<where>
<choose>
<when test="id!=null">id=#{id}</when>
<when test="name!=null">name=#{name}</when>
</choose>
</where>
</update>
5 foreach 标签语句
collection 表示迭代集合的名称,可以使用@Param注解指定,如下所示 该参数为必选
item 表示本次迭代获取的元素,若collection为List、Set或者数组,则表示其中的元素; 若collection为map,则代表key-value的value,该参数为必选
open 表示该语句以什么开始,最常用的是左括弧’(’,注意:mybatis会将该字符拼接到整体的sql语句之前,并且只拼接一次,该参数为可选项
close 表示该语句以什么结束,最常用的是右括弧’)’,注意:mybatis会将该字符拼接到整体的sql语句之后,该参数为可选项
separator mybatis会在每次迭代后给sql语句append上separator属性指定的字符,该参数为可选项
index 在list、Set和数组中,index表示当前迭代的位置,在map中,index代指是元素的key,该参数是可选项。
<!--
第一步:迭代集合,获取对应的item,和外部的(),拼接形成('zhangsan')
第二步:在之前的基础上拼接上逗号分隔符('zhangsan'),
第三步:继续迭代并拼接逗号 ('zhangsan'),('lisi'),
第四步:继续迭代并拼接逗号 ('zhangsan'),('lisi'),('wangwu')
-->
<foreach collection="list" item="item" separator=",">
(#{item})
</foreach>
<!--
第一步:拼接open指定的开始字符 (
第二步:迭代集合,拼接对应的item, ('zhangsan'
第三步:拼接separator指定的分隔符 ('zhangsan',
第四步:迭代集合,拼接对应的item, ('zhangsan','lisi'
第五步:拼接separator指定的分隔符('zhangsan','lisi',
第六步:拼接close指定的闭合字符 ('zhangsan','lisi','wangwu')
-->
<foreach collection="list" item="item" open="(" separator="," close=")">
#{item}
</foreach>
6 sql片段标签语句
在mapper.xml中有些sql片段,在多个sql语句中都有用,每个都写的话显得代码冗余,可以用到代码片段,将公共部分抽取出来
注意点:最好基于单表定义sql片段,并且sql片段中不要存在where标签,因为where标签会自动优化sql中的and和or
<sql id="choose-when-id-name">
<choose>
<when test="id!=null">id=#{id}</when>
<when test="name!=null">name=#{name}</when>
</choose>
</sql>
<update id="updatePwd">
update public."user"
<set>
<if test="name !=null">name=#{name},</if>
<if test="pwd !=null">pwd=#{pwd},</if>
</set>
<where>
<include refid="choose-when-id-name"></include>
</where>
</update>
7 trim标签语句
trim标记是一个格式化的标记,可以完成set或者是where标记的功能
1 用 trim 改写上面的 if+where 语句
prefix:前缀
prefixoverride:去掉第一个and或者是or
<select id="selectUserByUsernameAndSex" resultType="user" parameterType="com.harvey.java01.entity.User">
select * from user
<!-- <where>
<if test="username != null">
username=#{username}
</if>
<if test="sex != null">
and sex=#{sex}
</if>
</where> -->
<trim prefix="where" prefixOverrides="and | or">
<if test="username != null">
and username=#{username}
</if>
<if test="sex != null">
and sex=#{sex}
</if>
</trim>
</select>
2 用 trim 改写上面的 if+set 语句
suffix:后缀
suffixoverride:去掉最后一个逗号(也可以是其他的标记,就像是上面前缀中的and一样)
<!-- 根据 id 更新 user 表的数据 -->
<update id="updateUserById" parameterType="com.ys.po.User">
update user u
<!-- <set>
<if test="username != null and username != ''">
u.username = #{username},
</if>
<if test="sex != null and sex != ''">
u.sex = #{sex}
</if>
</set> -->
<trim prefix="set" suffixOverrides=",">
<if test="username != null and username != ''">
u.username = #{username},
</if>
<if test="sex != null and sex != ''">
u.sex = #{sex},
</if>
</trim>
where id=#{id}
</update>
2 mybatis分页
mybatis分页可以减少数据的处理量,分页的方式有两种:limit分页、PageHelper分页插件
1 limit分页方式
1 编写xxMapper.java
List<User> getUserByLimit(Map<String,Integer> data);
2 编写xxMapper.xml:
<select id="getUserByLimit" parameterType="map" resultType="user">
select * from usr limit #{currIndex},#{pageSize}
</select>
3 编写测试service层:
//接口
List<Usr> queryUserBySql(int currPage, int pageSize);
//实现类
public List<Student> queryUserBySql(int currPage, int pageSize) {
Map<String, Object> data = new HashedMap();
data.put("currIndex", (currPage-1)*pageSize);
data.put("pageSize", pageSize);
return xxMapper.getUserByLimit(data);
}
2 PageHelper分页插件
1 导入maven坐标
(由于使用了sql 解析工具,所以maven也引入了 jsqlparser.jar)
<dependency>
<groupId>com.github.pagehelper</groupId>
<artifactId>pagehelper</artifactId>
<version>4.0.0</version>
</dependency>
2 配置mybatis-config.xml拦截器插件
<plugins>
<plugin interceptor="com.github.pagehelper.PageHelper">
<property name="dialect" value="mysql"/>
</plugin>
</plugins>
配置拦截器时要知道新版拦截器是 com.github.pagehelper.PageInterceptor,而com.github.pagehelper.PageHelper 现在是一个特殊的 dialect 实现类,是分页插件的默认实现类,提供了和以前相同的用法。
设置数据库类型时有 Oracle,Mysql,MariaDB,SQLite,Hsqldb,PostgreSQL 六种数据库。
dialect默认情况下会使用 PageHelper 方式进行分页
1 PageHelper分页插件
- 只有紧跟在PageHelper.startPage方法后的第一个Mybatis的查询(Select)方法会被分页
- 对于带有for update的sql,会抛出运行时异常,对于这样的sql建议手动分页,毕 竟这样的sql需要重视在这里插入图片描述
- 分页插件不支持嵌套结果映射,由于嵌套结果方式会导致结果集被折叠,因此分页查询 的结果在折叠后总数会减少,所以无法保证分页结果数量正确。
3 Mybatis的缓存
1 一级缓存和二级缓存
1 MyBatis的一级缓存
一级缓存是SqlSession级别的,通过同一个SqlSession查询的数据会被缓存,下次查询相同的数据,就会从缓存中直接获取,不会从数据库重新访问
使一级缓存失效的四种情况:
- 不同的SqlSession对应不同的一级缓存。
- 同一个SqlSession但是查询条件不同。
- 同一个SqlSession两次查询期间执行了任何一次增删改操作。
- 同一个SqlSession两次查询期间手动清空了缓存。
sqlSession.clearCache()
2 MyBatis的二级缓存
- 二级缓存是SqlSessionFactory级别,通过同一个SqlSessionFactory创建的SqlSession查询的结果会被缓存;此后若再次执行相同的查询语句,结果就会从缓存中获取
- 二级缓存开启的条件
- 在核心配置文件中,设置全局配置属性cacheEnabled=“true”,默认为true,不需要设置
<settings>
<setting name="cacheEnabled" value="true"/>
</settings>
- 在Mapper.xml文件中添加cache标签
<cache/>
- 在需要进行缓存的查询语句上添加useCache属性
<select id="getUserById" resultMap="userResultMap" useCache="true">
SELECT * FROM user WHERE id = #{id}
</select>
- 二级缓存必须在SqlSession关闭或提交之后有效
- 查询的数据所转换的实体类类型必须实现序列化的接口
- 使二级缓存失效的情况:两次查询之间执行了任意的增删改,会使一级和二级缓存同时失效
- 手动清空缓存只对sqlSession一级缓存有效
2 MyBatis缓存查询的顺序
- 先查询二级缓存,因为二级缓存中可能会有其他程序已经查出来的数据,可以拿来直接使用
- 如果二级缓存没有命中,再查询一级缓存
- 如果一级缓存也没有命中,则查询数据库
- SqlSession关闭之后,一级缓存中的数据会写入二级缓存