9.1、环境搭建

9.1.1、创建新module

创建名为mybatis_dynamicSQL的新module,过程参考5.1节

9.1.2、创建Emp实体类

package org.rain.mybatis.pojo;

/**
* @author liaojy
* @date 2023/6/20 - 0:09
*/
public class Emp {
private Integer empId;
private String empName;
private Integer age;
private String gender; public Emp() {
} public Emp(Integer empId, String empName, Integer age, String gender) {
this.empId = empId;
this.empName = empName;
this.age = age;
this.gender = gender;
} public Integer getEmpId() {
return empId;
} public void setEmpId(Integer empId) {
this.empId = empId;
} public String getEmpName() {
return empName;
} public void setEmpName(String empName) {
this.empName = empName;
} public Integer getAge() {
return age;
} public void setAge(Integer age) {
this.age = age;
} public String getGender() {
return gender;
} public void setGender(String gender) {
this.gender = gender;
} @Override
public String toString() {
return "Emp{" +
"empId=" + empId +
", empName='" + empName + '\'' +
", age=" + age +
", gender='" + gender + '\'' +
'}';
}
}

9.1.3、创建Mapper接口和映射文件

++++++++++++++++++++++++++++分割线++++++++++++++++++++++++++++

<?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="org.rain.mybatis.mapper.DynamicSQLMapper"> </mapper>

9.1.4、创建测试类

9.2、if标签

9.2.1、接口方法

List<Emp> getEmpsByCondition(Emp emp);

9.2.2、映射文件

    <!--List<Emp> getEmpsByCondition(Emp emp);-->
<select id="getEmpsByCondition" resultType="Emp">
select * from t_emp where
<!--
if标签:通过test属性中的表达式判断标签中的内容是否会拼接到sql中
-->
<if test="empName != '' and empName != null">
emp_name = #{empName}
</if>
<if test="age != '' and age != null">
and age = #{age}
</if>
<if test="gender != '' and gender != null">
and gender = #{gender}
</if>
</select>

9.2.3、测试方法

模拟符合所有if标签的test属性中的表达式判断为true的情况

    @Test
public void testCetEmpsByCondition(){
SqlSession sqlSession = SqlSessionUtils.getSqlSession();
DynamicSQLMapper dynamicSQLMapper = sqlSession.getMapper(DynamicSQLMapper.class);
Emp emp = new Emp();
emp.setEmpName("张三");
emp.setAge(20);
emp.setGender("男");
List<Emp> emps = dynamicSQLMapper.getEmpsByCondition(emp);
System.out.println(emps);
sqlSession.close();
}

9.2.4、执行效果

所有if标签的内容都已顺利拼接到sql中

9.3、where标签

9.3.1、if标签缺陷

注意:当部分或全部if标签的test属性中的表达式判断为false时,sql可能会因为where条件的拼接报错

9.3.2、缺陷修正

9.3.2.1、方式一:增加恒成立条件

<!--List<Emp> getEmpsByCondition(Emp emp);-->
<select id="getEmpsByCondition" resultType="Emp">
select * from t_emp where 1=1
<if test="empName != '' and empName != null">
and emp_name = #{empName}
</if>
<if test="age != '' and age != null">
and age = #{age}
</if>
<if test="gender != '' and gender != null">
and gender = #{gender}
</if>
</select>

9.3.2.1、方式二:使用where标签

    <!--List<Emp> getEmpsByCondition(Emp emp);-->
<select id="getEmpsByCondition" resultType="Emp">
select * from t_emp
<!--
where标签的效果:
1、若标签中有条件成立,则自动生成where关键字,且能自动删除(前方)多余的and关键字
2、若标签中没有任何条件成立,则没有任何功能
-->
<where>
<if test="empName != '' and empName != null">
and emp_name = #{empName}
</if>
<if test="age != '' and age != null">
and age = #{age}
</if>
<if test="gender != '' and gender != null">
and gender = #{gender}
</if>
</where>
</select>

++++++++++++++++++++++++++++分割线++++++++++++++++++++++++++++

9.4、trim标签

    <!--List<Emp> getEmpsByCondition(Emp emp);-->
<select id="getEmpsByCondition" resultType="Emp">
select * from t_emp
<!--
trim标签:用于添加或去掉标签中的内容
prefix属性:在标签中内容的前面添加指定的内容
suffix属性:在标签中内容的后面添加指定的内容
prefixOverrides属性:在标签中内容的前面去掉指定的内容
suffixOverrides属性:在标签中内容的后面去掉指定的内容
-->
<trim prefix="where" suffixOverrides="and">
<if test="empName != '' and empName != null">
emp_name = #{empName} and
</if>
<if test="age != '' and age != null">
age = #{age} and
</if>
<if test="gender != '' and gender != null">
gender = #{gender} and
</if>
</trim>
</select>

9.5、choose标签组

    <!--List<Emp> getEmpsByCondition(Emp emp);-->
<select id="getEmpsByCondition" resultType="Emp">
select * from t_emp
<where>
<!--
choose、when和otherwise标签组合(少用):
when(相当于if...else)至少有一个,otherwise(相当于else)至多有一个
-->
<choose>
<when test="empName != '' and empName != null">
emp_name = #{empName}
</when>
<when test="age != '' and age != null">
age = #{age}
</when>
<when test="gender != '' and gender != null">
gender = #{gender}
</when>
</choose>
</where>
</select>

9.6、foreach标签

9.6.1、批量添加

9.6.1.1、接口方法

void insertMoreEmp(@Param("emps") List<Emp> emps);

9.6.1.2、映射文件

    <!--void insertMoreEmp(@Param("emps") List<Emp> emps);-->
<!--
foreach标签:
collection属性:设置要循环的数组或集合
item属性:设置表示数组或集合中迭代元素的变量名
separator属性:设置每次foreach循环之间的分隔符
-->
<insert id="insertMoreEmp">
insert into t_emp values
<foreach collection="emps" item="emp" separator=",">
(null ,#{emp.empName},#{emp.age},#{emp.gender},null)
</foreach>
</insert>

9.6.1.3、测试方法

    @Test
public void testInsertMoreEmp(){
SqlSession sqlSession = SqlSessionUtils.getSqlSession();
DynamicSQLMapper dynamicSQLMapper = sqlSession.getMapper(DynamicSQLMapper.class);
Emp emp1 = new Emp(null,"小明1",20,"男");
Emp emp2 = new Emp(null,"小明2",20,"男");
Emp emp3 = new Emp(null,"小明3",20,"男");
List<Emp> emps = Arrays.asList(emp1, emp2, emp3);
dynamicSQLMapper.insertMoreEmp(emps);
sqlSession.close();
}

9.6.1.4、执行效果

++++++++++++++++++++++++++++分割线++++++++++++++++++++++++++++

++++++++++++++++++++++++++++分割线++++++++++++++++++++++++++++

9.6.2、批量删除

9.6.2.1、接口方法

void deleteMoreEmp(@Param("empIds") Integer[] empIds);

9.6.2.2、映射文件

    <!--void deleteMoreEmp(@Param("empIds") Integer[] empIds);-->
<!--
foreach标签:
collection属性:设置要循环的数组或集合
item属性:设置表示数组或集合中迭代元素的变量名
separator属性:设置每次foreach循环之间的分隔符
open属性:设置foreach循环体的前缀
close属性:设置foreach循环体的后缀
-->
<delete id="deleteMoreEmp">
delete from t_emp where emp_id in
<foreach collection="empIds" item="empId" separator="," open="(" close=")">
#{empId}
</foreach>
</delete>

9.6.2.3、测试方法

    @Test
public void testDeleteMoreEmp(){
SqlSession sqlSession = SqlSessionUtils.getSqlSession();
DynamicSQLMapper dynamicSQLMapper = sqlSession.getMapper(DynamicSQLMapper.class);
Integer[] empIds = {6, 7};
dynamicSQLMapper.deleteMoreEmp(empIds);
sqlSession.close();
}

9.6.2.4、执行效果

++++++++++++++++++++++++++++分割线++++++++++++++++++++++++++++

++++++++++++++++++++++++++++分割线++++++++++++++++++++++++++++

9.6.2.5、映射文件(变体)

    <delete id="deleteMoreEmp">
delete from t_emp where
<foreach collection="empIds" item="empId" separator="or">
emp_id = #{empId}
</foreach>
</delete>

9.7、sql标签

    <!--
sql标签:设置一个sql片段
-->
<sql id="empColumns">
emp_id,emp_name,age,gender
</sql> <!--List<Emp> getEmpsByCondition(Emp emp);-->
<!--
include标签:
refid属性:通过sql标签的id引用相关sql片段
-->
<select id="getEmpsByCondition" resultType="Emp">
select <include refid="empColumns"></include> from t_emp
</select>

9、Mybatis之动态SQL的更多相关文章

  1. MyBatis的动态SQL详解

    MyBatis的动态SQL是基于OGNL表达式的,它可以帮助我们方便的在SQL语句中实现某些逻辑,本文详解mybatis的动态sql,需要的朋友可以参考下 MyBatis 的一个强大的特性之一通常是它 ...

  2. Mybatis解析动态sql原理分析

    前言 废话不多说,直接进入文章. 我们在使用mybatis的时候,会在xml中编写sql语句. 比如这段动态sql代码: <update id="update" parame ...

  3. mybatis 使用动态SQL

    RoleMapper.java public interface RoleMapper { public void add(Role role); public void update(Role ro ...

  4. MyBatis框架——动态SQL、缓存机制、逆向工程

    MyBatis框架--动态SQL.缓存机制.逆向工程 一.Dynamic SQL 为什么需要动态SQL?有时候需要根据实际传入的参数来动态的拼接SQL语句.最常用的就是:where和if标签 1.参考 ...

  5. 使用Mybatis实现动态SQL(一)

    使用Mybatis实现动态SQL 作者 : Stanley 罗昊 [转载请注明出处和署名,谢谢!] 写在前面:        *本章节适合有Mybatis基础者观看* 前置讲解 我现在写一个查询全部的 ...

  6. MyBatis探究-----动态SQL详解

    1.if标签 接口中方法:public List<Employee> getEmpsByEmpProperties(Employee employee); XML中:where 1=1必不 ...

  7. mybatis中的.xml文件总结——mybatis的动态sql

    resultMap resultType可以指定pojo将查询结果映射为pojo,但需要pojo的属性名和sql查询的列名一致方可映射成功. 如果sql查询字段名和pojo的属性名不一致,可以通过re ...

  8. mybatis.5.动态SQL

    1.动态SQL,解决关联sql字符串的问题,mybatis的动态sql基于OGNL表达式 if语句,在DeptMapper.xml增加如下语句; <select id="selectB ...

  9. MyBatis的动态SQL详解-各种标签使用

    MyBatis的动态SQL是基于OGNL表达式的,它可以帮助我们方便的在SQL语句中实现某些逻辑. MyBatis中用于实现动态SQL的元素主要有: if choose(when,otherwise) ...

  10. 利用MyBatis的动态SQL特性抽象统一SQL查询接口

    1. SQL查询的统一抽象 MyBatis制动动态SQL的构造,利用动态SQL和自定义的参数Bean抽象,可以将绝大部分SQL查询抽象为一个统一接口,查询参数使用一个自定义bean继承Map,使用映射 ...

随机推荐

  1. 2022-04-27:Alice 有一个下标从 0 开始的数组 arr ,由 n 个正整数组成。她会选择一个任意的 正整数 k 并按下述方式创建两个下标从 0 开始的新整数数组 lower 和 hig

    2022-04-27:Alice 有一个下标从 0 开始的数组 arr ,由 n 个正整数组成.她会选择一个任意的 正整数 k 并按下述方式创建两个下标从 0 开始的新整数数组 lower 和 hig ...

  2. Selenium - 快速上手之启动浏览器

    Selenium - 浏览器启动/关闭 使用webdriver前,需要先导入包 from selenium import webdriver,每次打开浏览器时; 执行完毕要记得关闭浏览器,使用 dri ...

  3. 用rust 写一个jar包 class冲突检测工具

    Rust很适合写命令行工具,特别是使用clap crate 更加方便,这篇文章介绍使用rust写一个jar包class冲突检测的工具.项目地址: https://github.com/Aitozi/j ...

  4. 计算机网络 ACL和ANT

    目录 一.ACL概况 二.ACL工作过程 三.ACL实验 四.ANT概况 五.ANT工作过程 六.ANT实验 一.ACL概况 概念:主要是对报文进行区分,路由器会对报文进行检查,查看是否符合通过标准或 ...

  5. computed与watch监听同一对象的场景

    今日项目需要一个详细的权限配置,如:路由权限,页面tabs权限,操作权限:而路由权限大都知道就不赘述,而操作权限这就涉及页面中每个按钮了,这里使用VUEX进行管理 1.配置store store 中 ...

  6. Odoo 13之十三 :开发之创建网站前端功能

    Odoo 13开发之创建网站前端功能 Odoo 起初是一个后台系统,但很快就有了前端界面的需求.早期基于后台界面的门户界面不够灵活并且对移动端不友好.为解决这一问题,Odoo 引入了新的网站功能,为系 ...

  7. Django 有关 models 数据类型介绍:

    在model中添加字段的格式一般为:  field_name = field_type(**field_options) 一  field options(所有字段共用) 1  null   默认为F ...

  8. 在C++中,传值还是传引用?

    情况一:需要修改原对象 需要修改原对象的情况,必须要传引用.这种情况没什么要说的. 情况二:不需要修改原对象 对于内置类型(整数.浮点数.字符类型等),传值效率更高.主要有三点原因: 内存开销更小.由 ...

  9. 蜂窝移动通信(IOT)接入流程

      蜂窝物联网   蜂窝物联网(Cellular IoT)就是使用现有的蜂窝网络连接物联网设备而形成的物联网,是一种将物理设备与互联网连接起来的方式.通过蜂窝物联网,人们将一些物理设备--如传感器-- ...

  10. 可视化生信分析利器 Galaxy 之 Docker 开发

    1. 背景 我们常常会基于某个 image 来启动一个 container,在这个 container 中我们可能会执行某些操作,比如创建一个文件,但是当这个 container 退出以后,如果我们以 ...