前言
我们知道在项目开发中之前使用数据库查询,都是基于jdbc
,进行连接查询,然后是高级一点jdbcTemplate
进行查询,但是我们发现还是不是很方便,有大量重复sql语句,与代码偶合,效率低下,于是就衍生出来ORM
框架,如Mybatis,Hibernate,
还有SpringBoot的,Spring Data JPA
条件查询
我们知道在mybatis mapper
文件中条件查询符,如>=,<,
之类是不能直接写的会报错的需要转移一下 如下图表
详细内容参考
常见的条件查询操作有
我们通过mybatis 提供的特有标签进行条件判断,达到动态拼接sql
语句
if标签 where标签 choose when otherwise标签 foreach标签
快速入门
if标签
语法:
<if test="xxx != null and xxx != ''">
test中写判断条件 参数直接paramN或者别名 多个条件使用and
或者or
连接
只要条件成立就拼接在Sql语句中,都成立就全部都拼接
注意where子句中加上1=1来规避and的风险
如下例子:
<select id="selg" resultType="log">
select * from log where 1=1
<if test="param1!=null and param1!=''">
and outno=#{param1}
</if>
<if test="param2!=null and param2!=''">
and inno=#{param2}
</if>
</select>
where标签
对上面if
标签条件判断where
连接做了处理会自动的给Sql
语句添加where
关键字,并将第一个and
去除
上面sql可以改造成如下:
<select id="selg" resultType="log">
select * from log
<where>
<if test="param1!=null and param1!=''">
and outno=#{param1}
</if>
<if test="param2!=null and param2!=''">
and inno=#{param2}
</if>
</where>
</select>
choose when otherwise标签
类似于Java
语法中的,case,switch
语句判断
条件只要有一个成立,其他的就不会再判断了。如果没有成立的条件则默认执行otherwise
中的内容
上面sql可以改造成如下:
<select id="selg" resultType="log">
select * from log
<where>
<choose>
<when test="param1!=null and param1!=''">
and outno=#{param1}
</when>
<when test="param2!=null and param2!=''">
and inno=#{param2}
</when>
<otherwise>
and 1=1
</otherwise>
</choose>
</where>
</select>
foreach标签
语法:
<foreach collection="idList" item="id" open="(" separator="," close=")">
</foreach>
- collection:要遍历的集合对象
- item:记录每次遍历的结果
- open:在结果的左边添加内容
- separator:结果和结果之间的内容
- close:在最后添加的内容
常用于in
查询,和批量插入
操作 如下案例:
<select id="selF" parameterType="list" resultType="account">
select * from account where ano in
<foreach collection="list" item="item" open="(" separator="," close=")">
#{item}
</foreach>
</select>
<insert id="insertBatch">
INSERT INTO t_user
(id, name, password)
VALUES
<foreach collection ="userList" item="user" separator =",">
(#{user.id}, #{user.name}, #{user.password})
</foreach >
</insert>
其他标签使用参考点击进入·
场景案例
- 当我们需要对多张表的关联数据进行复杂动态条件查询的时候,就需要用到
if
标签进行判断 如下
根据用户手机号姓名年龄性别,等进行动态条件检索,这个时候我们需要动态通过调节去拼接sql
当条件满足sql语句加上对应条件差许
<select id="findUsersByUser" resultType="cn.soboys.kmall.sys.entity.User">
select tu.USER_ID,tu.USERNAME,tu.SSEX,td.DEPT_NAME,tu.MOBILE,tu.EMAIL,tu.STATUS,tu.CREATE_TIME,
td.DEPT_ID
from t_user tu left join t_dept td on tu.DEPT_ID = td.DEPT_ID
where tu.ADMIN_TYPE_ID >= 0 AND tu.ADMIN_TYPE_ID <= #{userParams.adminType}
<if test="userParams.roleId != null and userParams.roleId != ''">
and (select group_concat(ur.ROLE_ID)
from t_user u
right join t_user_role ur on ur.USER_ID = u.USER_ID,
t_role r
where r.ROLE_ID = ur.ROLE_ID
and u.USER_ID = tu.USER_ID and r.ROLE_ID=#{userParams.roleId})
</if>
<if test="userParams.mobile != null and userParams.mobile != ''">
AND tu.MOBILE =#{userParams.mobile}
</if>
<if test="userParams.username != null and userParams.username != ''">
AND tu.USERNAME like CONCAT('%',#{userParams.username},'%')
</if>
<if test="userParams.ssex != null and userParams.ssex != ''">
AND tu.SSEX =#{userParams.ssex}
</if>
<if test="userParams.status != null and userParams.status != ''">
AND tu.STATUS =#{userParams.status}
</if>
<if test="userParams.deptId != null and userParams.deptId != ''">
AND td.DEPT_ID =#{userParams.deptId}
</if>
<if test="userParams.createTime != null and userParams.createTime != ''">
AND DATE_FORMAT(tu.CREATE_TIME,'%Y%m%d') BETWEEN substring_index(#{userParams.createTime},'#',1) and substring_index(#{userParams.createTime},'#',-1)
</if>
</select>
对应mapper对应的方法
<T> IPage<User> findUsersByUser(Page<T> page, @Param("userParams") SearchUserParams userParams);
对应参数实体对象
@Data
public class SearchUserParams {
private String username;
private String mobile;
private String status;
private String ssex;
private Long deptId;
private String createTime;
private long adminType;
private String roleId;
}
通过if
标签去判断条件是否满足,满足就拼接对应sql
注意在上面我们提到的条件拼接第一个是
where
连接,而不是and
应规避and风险保证sql语法正确 如下
<select id="findSearchCouponsPage" parameterType="cn.soboys.kmall.bean.web.params.SearchCouponParams" resultType="coupon">
select *
from coupon c
left join user_coupon uc on c.coupon_id = uc.coupon_id
WHERE 1 = 1
<if test="couponParams.userId != null and couponParams.userId != ''">
and uc.user_id =#{couponParams.userId}
</if>
<if test="couponParams.status != null and couponParams.status != ''">
and c.status =#{couponParams.status}
</if>
<if test="couponParams.couponId != null and couponParams.couponId != ''">
and c.coupon_id =#{couponParams.couponId}
</if>
<if test="couponParams.couponType != null and couponParams.couponType != ''">
and c.type =#{couponParams.couponType}
</if>
</select>
我们可以通过假定给他一个默认条件 WHERE 1 = 1
来解决,也可以通过嵌套where
标签来解决