Mybatis 动态sql的编写|开启二级缓存

news2024/11/15 8:49:30

❤️作者主页:微凉秋意
✅作者简介:后端领域优质创作者🏆,CSDN内容合伙人🏆,阿里云专家博主🏆
✨精品专栏:数据结构与课程设计
🔥系列专栏:javaweb

文章目录

  • 前言
  • 动态SQL
    • 模糊查询
    • where if 标签
    • update set 标签
    • foreach 标签
    • useGeneratedKeys 获取插入数据的主键值
  • Mybatis 缓存
    • 一级缓存
    • 二级缓存

前言

上篇博文把表连接查询和三种对应关系的写法记录总结了,本篇要把 mybatis 中的动态sql 的使用以及缓存知识记录下来。

动态SQL

在解释 where if 标签之前先进行一个模糊查询的操作。

模糊查询

如下面一张表:

在这里插入图片描述
查询所有 姓人员的信息:

mapper接口中定义方法:

List<Person> selectName(String name);

xml 中编写 sql 语句:

<select id="selectName" resultMap="personMap">
        select *
        from t_person
        where person_name like concat('%', #{name}, '%');
    </select>

这里的 concat('%', #{name}, '%') 写法使用了字符串拼接的技巧,这样在查询语句里就是 %name% 的模糊查询了。
如果这样写:'%#{name}%' 是不行的,#{}会失效。

编写测试类:

在这里插入图片描述
测试结果:
在这里插入图片描述
可以看到所有 姓的人信息被查询到了。

where if 标签

姓李的太多,那么我们可以加一个年龄范围,于是需要增加两个参数:

在这里插入图片描述

这里要注意,参数在两个以上需要使用 注解 来绑定。

重点来看 xml中的 sql 语句

<select id="selectContent" resultMap="personMap">
        select * from t_person
        <where>
            <if test="name != null and name !=''">
                person_name like concat('%',#{name},'%')
            </if>
            <if test="ageMin > 0">
                and age &gt;= #{ageMin}
            </if>
            <if test="ageMax > ageMin">
                and age &lt;= #{ageMax}
            </if>
        </where>

    </select>
  • 可以看到这里的 where 关键字变成了一个标签 <where>,这是因为:
    • 如果这三个参数都不填写的话,where 下的条件就不会起作用,
      此条 sql 就变成了查询所有。
    • 显然,查询所有的语法里是没有 where 关键字的,因此使用标签代替。
    • 除了使用 <where> 标签之外,也可以使用类似 1=1and 的方法拼接。
      • 比如: select * from t_person where 1=1 and ...
  • 此外,where 标签还可以去除多余的 and 前缀。
  • if 标签用来做判断,test 里写判断的条件,标签内写判断成功后的 sql 语句:
    • 这里就是判断名字是否为空以及年龄的取值范围是否合理:
      • test 里面的符号正常写,andor 代表并且和或者。
      • sql 语句里,&gt; 是大于的意思,&lt; 是小于的意思。

编写测试类:

在这里插入图片描述

测试结果:
在这里插入图片描述
很明显,年龄在 20~28 且姓李的只有 “李白” 自己。

update set 标签

以往的修改操作都是将对象的所有属性都修改或者重写一遍,这样是不合逻辑的,因此可以在修改的 sql 语句里使用 if 标签来优化。

mapper 接口中写方法:

void update(Person person);

xml 中编写 sql:

	<update id="update">
        update t_person
        <set>
            <if test="personName !=null and personName !=''">
                person_name = #{personName},
            </if>
            <if test="age !=null">
                age         = #{age},
            </if>
            <if test="birthday != null and birthday!= ''">
                birthday    = #{birthday},
            </if>
            <if test=" sex !=null and sex !=''">
                sex = #{sex},
            </if>
            <if test="mobile != null and mobile !=''">
                mobile = #{mobile},
            </if>
            <if test="address !=null and address!=''">
                address = #{address}
            </if>
        </set>
        where person_id = #{personId}
    </update>
  • if 标签想必好理解,但是set 标签是做什么的呢?原因是:
    • 假如只修改名字,那么 sql 语句将变为:
      update t_person set person_name = ?, where ...
      
    • 可以发现?占位符后面会有多余的逗号导致 sql 语法错误
    • <set> 标签会自动去除多余的逗号后缀。

值得注意的是:set 标签去除,后缀,where 标签去除and前缀。

foreach 标签

相信大家都知道foreach 跟循环有关系,没错,我们可以利用此标签来完成 批量删除 操作。

从 sql 的语句来看,批量删除应该这样写:

delete from t_person where person_id in (?,?,?,...)

那么 in 后的形式该怎么用 mybatis 表达出来呢,有括号,不定数量的逗号…

不急,一步步来,先写 mapper接口:

void deleteMany(@Param("myIds") int[] ids);

再编写 sql 语句:

	<delete id="deleteMany">
        delete from t_person where person_id in
        <foreach collection="myIds" open="(" close=")" separator="," item="id">
            #{id}
        </foreach>
    </delete>

利用 foreach 就可以将参数集合中的值依次放入到 in 后的括号内:(id,id,id...)

foreach 标签相关解释:

  • collection:被循环的数组或者集合,注意如果是参数必须使用@Param 起名
  • open : 循环以什么开始
  • close :循环以什么结束
  • separator: 循环以什么分割
  • item : 每次循环的元素

编写测试类(不要忘记手动提交事务):

在这里插入图片描述

测试结果:

在这里插入图片描述
可以看到数组里的元素依次放入了括号内,并成功做到了批量删除。

useGeneratedKeys 获取插入数据的主键值

一般我们习惯将表中的主键设为自增长,那么在使用 insert 插入数据时,主键是不用赋值的,而是用 null 代替。那怎么才能在插入后直接得到该数据的主键值呢?那就使用 useGeneratedKeys

来看具体的使用:

<insert id="insert" useGeneratedKeys="true" keyProperty="personId" keyColumn="person_id">
    insert into t_person
    values (null, #{personName}, #{age}, #{sex}, #{birthday}, #{mobile}, #{address})
</insert>

将值设为 true 后,keyPropetry 的值填写属性名,keyColumn 的值填写表的字段名。

编写测试类:

	@Test
    public void insert(){
    	SqlSession sqlSession = MybatisUtil.getSqlSession();
    	PersonMapper mapper = sqlSession.getMapper(PersonMapper.class);
        Person person = new Person(null,"微凉",21,"1",new Date(),"1359356123","爱琴海");
        mapper.insert(person);
        sqlSession.commit();
        MybatisUtil.closeSqlSession(sqlSession);
        System.out.println(person);
    }

测试结果:

在这里插入图片描述
可以看到插入数据后的主键值为 240,这便是自动获取的结果。

Mybatis 缓存

有关 IO 操作一般都会很耗时,在一个 javaweb 项目中,页面到 tomcat 与 tomcat 到数据库都属于IO操作。因此 mybatis 在 tomcat 中创建了一块叫做缓存的空间,在缓存中保存比如查询操作的数据,以后进行重复操作时,就不必到数据库中获取了,而是直接在 tomcat 内部获取。

如果对数据库进行修改,缓存中的数据跟数据库中的对应不上,那么此时缓存中的就变成了脏数据

一级缓存

特点:

  • 默认开启
  • 作用在同一个 SqlSession
  • 脏数据的处理: 一旦执行增删改,立刻清空缓存

但是一级缓存并没有什么意义,因为作用范围的缘故,它只能在一个 SqlSession ,也就是一个事务内,一旦提交事务,缓存就会被清空,没有实战意义。

二级缓存

二级缓存的作用范围大,符合实战的需求,该如何开启:

  1. 实体类需要实现可序列化接口:implements Serializable
  2. mybatis-config.xml 文件中添加<settings> 标签:
    <!--开启二级缓存 settings 写在 properties 和 typeAliases 之间-->
    <settings>
       <setting name="cacheEnabled" value="true"/>
    </settings>
    
  3. 在对应 mapperxml 文件中添加 <cache> 标签:
    <cache size="1024" eviction="LRU" flushInterval="60000"/>
    
    • size:定义二级缓存空间大小 (单位是 M)
    • eviction:定义缓存到达上限时的淘汰策略,常使用LRU算法:
      • 叫做最近最久未使用算法,即清空最近使用次数最少的缓存
    • flushInterval:定义清空缓存的间隔时间 (单位 ms)

作用范围:

  • 同一个sqlSessionFactory内,而且必须提交事务。

脏数据的处理:

  1. 一旦执行增删改,默认清空同一个 namespace 下的二级缓存。
  2. 自定义清空缓存的策略:
    • flushCache属性,值为 true 时执行清空缓存,为 false 则不清空。
    • insertdeleteupdate 标签里默认为 true
    • select 标签里默认为 false

脏数据并不都是一定要清空,因此在增删改查标签内可以自由的设定缓存清空的策略。


mybatis 框架的记录到此结束,一共三篇,感兴趣的小伙伴可以订阅此专栏。

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

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

相关文章

Linux-主要目录

/: 根目录&#xff0c;一般根目录下只存放目录&#xff0c;在Linux下有且只有一个根目录&#xff0c;所有的东西都是从这里开始当在终端里输入 /home,其实是在告诉电脑&#xff0c;先从/&#xff08;根目录&#xff09;开始&#xff0c;再进入到home目录/bin、/usr/bin:可执行二…

java基于ssm的旅游景点门票预订网站

通过本系统&#xff0c;能够实现用户进行登录后&#xff0c;在网站上对旅游景点信息进行门票预订&#xff0c;也可以通过在线留言系统&#xff0c;了解景点相关信息&#xff0c;为用户提供全方位的服务&#xff0c;同时为节省用户的时间&#xff0c;本系统采用分地名的方式&…

2022尚硅谷SSM框架跟学(七)Spring MVC基础二

2022尚硅谷SSM框架跟学 七 Spring MVC基础二4.SpringMVC获取请求参数4.1.通过ServletAPI获取4.2.通过控制器方法的形参获取请求参数4.3.RequestParam4.4.RequestHeader4.5.CookieValue4.6.通过POJO获取请求参数4.7.解决获取请求参数的乱码问题5.域对象共享数据5.1.使用ServletA…

LCX端⼝转发

LCX是一款十分经典的内网端口转发工具&#xff0c;基于Socket套接字&#xff0c;具有端口转发和端口映射的功能。但是目前很多杀软已经将其加入了特征库&#xff0c;在实际利用的时候需要自行做免杀处理。lcx有三个功能&#xff1a;第一个功能将本地端口转发到本地另一个端口上…

红米10A 一键root教程 跳过 秒过 168小时解锁BL 限制完美面具root保数据救黑砖刷机 解决无法解锁BL 新版本问题

新版本红米10A 无法秒解锁BL。root bootloader 要刷机、线刷救砖、获取Root权限都需要解除bootloader锁 我们称解除刷机限制的这个操作就叫解锁BL&#xff0c;只有解锁BL后&#xff0c;才可以进行刷机、刷Root权限等等一些列的操作。&#xff08;注&#xff1a;bootloader锁不…

Spring为什么这么火 之 五大类注解

目录 前言 存储Bean对象 1、配置扫描路径 2、将Bean存储到容器中 2.1、5大注解实现&#xff08;Controller、Service、Repository、Component、Configuration) Controller Service Repository Configuration Component 为什么需要五个类注解&#xff1f; 五大类注解的关…

Jmeter分布式

Jmeter的分布式控制&#xff0c;就是指远程启动功能。执行原理&#xff1a; 1、一台作为调度机(master)&#xff0c;其它机器做为执行机(slave) 2、执行时&#xff0c;master会把脚本发送到每台slave上&#xff0c;slave 拿到脚本后就开始执行 3、执行完成后&#xff0c;sla…

go语言编译文件踩坑

目录 1 no required module provides package xxx go.mod file not found in current directory or any parent directory; see go help modules 方式1 执行 go env -w GO111MODULEauto后再编译 方式2 创建go.mod 2 报错syscall.go:84:16: undefined: unsafe.Slice 3 b…

ArrayList集合类和对象数组案例区别讲解

目录 对象数组案例讲解 集合 ArrayList类的构造方法和成员方法 对象数组案例讲解 创建一个学生数组&#xff0c;存储三个学生对象并遍历 package com.demo01; /*** * author 无限嚣张菜菜* 分析&#xff1a;* A&#xff1a;定义学生类* B&#xff1a;创建学生数组* C&am…

关于PKI 你了解多少

公钥基础设施PKI&#xff08;Public Key Infrastructure&#xff09;&#xff0c;是利用公开密钥机制建立起来的基础设施&#xff0c;在信息互联互通时代&#xff0c;可以说它是创建、颁发、管理、注销公钥证书所涉及的所有软件、硬件集合体。对拥有庞大用户的系统来说&#xf…

Activity的5种启动模式详解(新增singleInstancePerTask类型)

前言 看到这个标题的时候&#xff0c;也许你会认为我写错了&#xff0c;Activity不是只有四种启动模式吗&#xff1f;分别为startard,singleTop,singleTask,singleInstance这四种。 一般来说是四种&#xff0c;但是android12的时候新加入了singleInstancePerTask类型&#xf…

【金三银四系列】之MySql面试突击(2023版)

Mysql金三银四面试突击班 1.Bin log是什么&#xff0c;有什么用&#xff1f;&#xff08;数据库被人干掉了怎么办&#xff1f;&#xff09; 1.bin Log: 数据恢复 主从复制 MySQL Server 层也有一个日志文件&#xff0c;叫做 binlog&#xff0c;它可以被所有的存储引擎使用。…

PRD-产品需求说明文档

产品需求说明文档&#xff08;PRD&#xff09;可以将产品设计思路清晰的展现给团队人员&#xff0c;便于他们快速理解产品。产品需求说明文档如何写呢&#xff1f;本文希望能够得到高人指点&#xff0c;本人是小白&#xff0c;同时不建议其他小白参考此文以免造成干扰。 包括文…

Spark 学习案例

案例1&#xff1a;搜索引擎日志分析 数据来源&#xff1a;使用搜狗实验室提供的【用户查询日志】数据。使用Saprk框架&#xff0c;将数据封装到RDD中进行数据处理分析。 数据网址&#xff1a;数据地址 这个地址可能过期了&#xff0c;需要的伙伴可以私聊博主。 数据格式&…

css清除浮动的方法

浮动的盒子会脱离标准流&#xff0c;不占有自己原先的位置&#xff0c;导致下面的其他标签往上移动 此时&#xff0c;可以给浮动的盒子添加一个父盒子&#xff08;块级&#xff09;&#xff0c;并设置高度&#xff0c;就可以避免下面的其他标签上浮 但是有时候父盒子不能设置高…

pycharm:新建虚拟环境和安装依赖

前言 小编深有体会&#xff0c;在刚开始用pycharm跑python的项目的时候&#xff0c;一时间不知道如何下手&#xff0c;特别是作为一个新手小白&#xff0c;这里总结了一份新手避坑指南&#xff0c;主要是新建虚拟环境&#xff08;生成一个项目对应的解释器&#xff09;以及安装…

温度预测 python | 使用 Python 可以使用机器学习模型进行温度预测

使用 Python 可以使用机器学习模型进行温度预测。常用的模型有回归分析、随机森林等。使用前需要准备足够的历史数据并进行特征工程&#xff0c;构建模型并进行训练&#xff0c;最后使用预测结果。 文章目录温度预测 回归分析导入必要的库&#xff1a;读取温度数据&#xff1a;…

Java中的插入排序和希尔排序

插入排序&&希尔排序插入排序希尔排序上一篇博客我给大家伙说了一下子堆排序,之所以我把插入排序和希尔排序放在一起呢,是因为希尔排序实际上用到了插入排序的思想,希望下面的内容能够帮助到大家.对于插入排序呢,我们可以参考抓牌顺牌,就在一般情况下,我们也不考虑什么组…

对象在Eden区分配

一、对象在Eden区分配大多数情况下&#xff0c;对象在新生代中 Eden 区分配。当 Eden 区没有足够空间进行分配时&#xff0c;虚拟机将发起一次Minor GC。在测试之前我们先来看看 Minor GC和Full GC 有什么不同呢&#xff1f;Minor GC/Young GC&#xff1a;指发生新生代的的垃圾…

oracle11g SAP测试机归档日志暴增排查(二)

接上面一的内容&#xff0c;通过logminer可以知道是因为oracle11g设置awr快照引起的插入数据&#xff0c;所以要看这个插入是否正常。 之前也发现SYSAUX表空间也没有多少了&#xff0c;应该这个原因引起产生大量的日志 6、查找SYSAUX表空间满的原因 对于SYSAUX表空间而言&…