MyBatis注解实现动态SQL问题
作者:杨哥学编程 发布时间:2023-08-19 02:34:09
标签:MyBatis,注解,动态SQL
MyBatis注解实现动态SQL
在 Mybatis 中,使用注解可以很方便的进行sql操作,但很多动态 SQL 都是由 xml 配置实现的。
而随着 SpringBoot的逐渐发展,越来越多的配置由配置文件转成注解的形式。其中包括动态 SQL 。
Mybatis 的注解中,使用太过长的sql语句看起来很不美观。
@Select("SELECT title from book_tbl where id = #{id}")
如果想要在 mapper 中进行非必要关键字的查询时就需要使用动态 SQL,与 xml 配置不同的是,@Select注解中 SQL 语句必须以 <script> 标签包裹。
@Select("<script>"+
"SELECT id " +
"from book_tbl" +
"<where>" +
"<if test 'company_id != null'>" +
"and company_id = #{company_id}" +
"</if>" +
"<if test 'title != null'>" +
"and title like CONCAT('%',#{title},'%')" +
"</if>" +
"</where>" +
"ORDER BY create_time desc,rank desc" +
"</script>")
MyBatis动态拼接 SQL参数
之前用JPA可以在@Query注解上拼接sql,实现动态查询;现在用mybatis,如何实现sql的动态拼接参数
举例,在JPA中可以实现类似于下面的sql拼接
plateNumber2 = plateNumber2 + "," + plateNumber;
String[] split = plateNumber2.split(",");
StringBuffer stringBuffer = new StringBuffer();
stringBuffer.append("SELECT plate_number from p_park_record where park_key = '"+parkDO.getParkKey()+"'");
stringBuffer.append(" and (");
for (int i = 0; i < split.length; i++) {
stringBuffer.append(" plate_number = "+split[i]);
if (i != split.length-1){
stringBuffer.append(" OR ");
}
}
stringBuffer.append(" )");
stringBuffer.append(" order by sensor_exit_time desc");
stringBuffer.append(" limit 1");
此代码中,核心问题是for循环,将不确定数量的同一个参数做拼接
那么在mybatis中该如何实现
mybatis实现:
plateNumber2 = plateNumber2 + "," + plateNumber;
ParkRecordDO parkRecordDO = new ParkRecordDO();
parkRecordDO.setPlateNumber(plateNumber2);
parkRecordDO.setParkKey(parkDO.getParkKey());
String onlyPlateNum = parkRecordMapper.getOnlyPlateNum(parkRecordDO);
<select id="getOnlyPlateNum" parameterType="ParkRecordDO" resultType="java.lang.String">
SELECT plate_number from p_park_record where park_key = #{parkKey} and sensor_exit_time is not null and
<foreach collection="plateNumber.split(',')" index="index" item="id" open="(" separator=" or " close=")">
plate_number = #{id}
</foreach>
order by sensor_exit_time desc
limit 1
</select>
如此实现,与JPA拼接效果相同,重点在于“<foreach”标签中separator属性,该属性为元素之间的分隔符,将分隔符设置为“or”,以此来实现该程序逻辑的拼接
最后补充几个知识点
foreach标签的使用
<!--
foreach
collection:指定要遍历的集合
list类型的参数会特殊处理封装在map中,map的key就叫list
item:将当前遍历出的元素赋值给指定的变量
separator:元素之间的分割符
open: 遍历出的所有结果拼接一个开始字符
close:遍历出的所有结果拼接一个结束字符
index:索引,遍历list的时候是索引,item就是当前值;
遍历map时index标识的就是map的key,item就是map的值
#{变量名}: 就能取出变量的值就是当前遍历出的元素
-->
<select id="queryUserByForEach" resultType="user">
select * from easybuy_user where id in
<foreach collection="ids" item="item_id" separator=","
open="(" close=")">
#{item_id}
</foreach>
</select>
<insert id="batchSaveUser" parameterType="user">
insert into easybuy_user(loginName, userName, password) values
<foreach collection="userList" item="user" separator=",">
(#{user.loginName}, #{user.userName}, #{user.password})
</foreach>
</insert>
set标签的使用
<!--
使用set标签或者trim标签与if标签相结合
实现动态更新sql语句的拼接
-->
<update id="updateUserByCondition" parameterType="user">
update easybuy_user
<set>
<if test="userName != null">
username = #{userName},
</if>
<if test="email != null">
email = #{email},
</if>
</set>
where id = 26;
</update>
choose和when标签的使用(作用类似与java中的switch-case)
<select id="queryUserByChoose" resultType="com.unistart.entity.User">
select * from easybuy_user
<where>
<choose>
<when test="id != null">
id=#{id}
</when>
<when test="userName != null">
userName like #{userName}
</when>
<when test="sex != null">
sex = #{sex}
</when>
<otherwise>
1=1;
</otherwise>
</choose>
</where>
</select>
trim标签的使用
<!--
trim标签的使用:解决后面多出的and或or
prefix="":前缀,trim标签体中是整个字符串拼串后的结果
prefix给拼串后的整个字符串加一个前缀
prefixOverrides="":前缀覆盖:去掉整个字符串前面多余的字符
suffix="":后缀,给拼串后的整个字符串加一个后缀
suffixOverrides="":后缀覆盖,去掉整个字符串后面多余的字符
-->
<select id="queryUserByTrim" resultType="user">
select * from easybuy_user
<trim prefix="where" suffixOverrides="and">
<if test="id != null">
id = #{id} and
</if>
<if test="userName != null and userName!=""">
userName like #{userName} and
</if>
<if test="email != null and email!=""">
email = #{email} and
</if>
</trim>
</select>
where标签的使用
<!--
查询用户:要求,携带哪些字段查询条件就按这些字段进行查询
使用的OGNL,类似与EL表达式
从参数中取值进行判断,遇见特殊符号使用转义字符
使用where标签时要注意,它只能去除第一个多出来的and或or
-->
<select id="queryUserByCondition" resultType="user">
select * from easybuy_user
<where>
<if test="id != null">
and id = #{id}
</if>
<if test="userName != null and userName!=""">
and userName like #{userName}
</if>
<if test="email != null and email!=""">
and email = #{email}
</if>
</where>
</select>
来源:https://blog.csdn.net/qq_47183158/article/details/123439717
0
投稿
猜你喜欢
- 1. selectKey标签查询DDLCREATE TABLE `luck_reward_info` ( `id` int NO
- Spring BeanPostProcessor执行顺序首先 Spring 通过调用构造方法创建 User 对象;User 对象创建好之后,
- 策略模式的应用场景策略模式是否要使用,取决于业务场景是否符合,有没有必要。是否符合如果业务是处于不同的场景时,采取不同的处理方式的话,就满足
- 本文介绍了Spring Boot + MyBatis读写分离,有需要了解Spring+MyBatis读写分离的朋友可参考。希望此文章对各位有
- instanceof 严格来说是Java中的一个双目运算符,用来测试一个对象是否为一个类的实例,用法为:boolean result = o
- 目录什么是异常?编译时还是运行时?“受检异常”究竟可不可取?我的观点什么是异常?要了解受检异常,首先要了解什么是异常。在Java中,异常是一
- 一:算术运算符1.算术运算符有哪些①基本四则运算符:+ - * / %②增量赋值运算符:+= -= *= /= %=③自增/自减运算符++
- 1|1简介最近基于最新的Activiti7配置了SpringBoot2。简单上手使用了一番。发现市面上解决Activiti7的教程很少,采坑
- 在编写ui界面时因为手机分辨率大小不同,所以展现出来的效果也是不同的,这个时候就需要考虑适配器,让根据手机分辨率自动适配相应尺寸来展示界面,
- 前言CyclicBarrier和CountDownLatch这两个工具都是在java.util.concurrent包下,并且平时很多场景都
- 星期天小哼和小哈约在一起玩桌游,他们正在玩一个非常古怪的扑克游戏——“小猫钓鱼”。游戏的规则是这样的:将一副扑克牌平均分成两份,每人拿一份。
- 最近公司需要做一个告警页面的功能,需要分页,查了很多资料发现PageHelper比较合适故写一篇从零开始的PageHelper使用的教程,也
- 简介Lambda表达式是Java SE 8中一个重要的新特性。lambda表达式允许你通过表达式来代替功能接口。 lambda表达式就和方法
- 项目里一直用的是 spring-security ,不得不说,spring-security 真是东西太多了,学习难度太大(可能我比较菜),
- 简单工厂简单工厂模式是属于创建型模式,是工厂模式的一种。简单工厂模式是由一个工厂对象决定创建出哪一种产品类的实例。定义了一个创建对象的类,由
- 本文实例为大家分享了java将一个目录下的所有数据复制到另一个目录下的具体代码,供大家参考,具体内容如下/* 将"C:\\Java
- 1、首先创建一个测试实体类Person,并携带如上注解,其注解的作用描述在messagepackage com.clickpaas.pojo
- 环境Linux版本:CentOS 6.5、Ubuntu 12.04.5 JDK版本:JDK 1.7目录方法一:手动解压JDK的压缩包,然后设
- 一、需求对于Java开发工程师来说,可能手头上同时负责不同的项目,但是由于历史的原因,Java版本可能没有做到统一升级,有的项目是使用JDK
- 最近在做一个需求:从其他系统的ftp目录下载存储图片url的文件,然后读取文件中的url地址,根据地址下载图片后按天压缩成一个包,平均一个地