0


【MyBatis框架】动态SQL

MyBatis之动态SQL

目录

开发人员在使用JDBC或者其他类似的框架进行数据库开发时,通常都要根据需求去手动拼装SQL,这是一个非常麻烦且痛苦的工作,而MyBatis提供的对SQL语句动态组装的功能,恰能很好的解决这一麻烦工作。

动态SQL是MyBatis的强大特性之一,其主要元素如下:
元素说明< if >判断语句,用于条件单分支判断< choose >相当于Java中的switch语句,用于多分支判断< where >,< trim >,< set >辅助元素,用于处理一些SQL的拼装,特殊字符等问题< foreach >循环语句,常用于in语句等列举条件< bind >用于模糊查询
实体类

publicclassEmp{privateInteger empId;privateString empName;privateInteger age;privateString gender;publicEmp(){}publicEmp(Integer empId,String empName,Integer age,String gender){this.empId = empId;this.empName = empName;this.age = age;this.gender = gender;}publicIntegergetEmpId(){return empId;}publicvoidsetEmpId(Integer empId){this.empId = empId;}publicStringgetEmpName(){return empName;}publicvoidsetEmpName(String empName){this.empName = empName;}publicIntegergetAge(){return age;}publicvoidsetAge(Integer age){this.age = age;}publicStringgetGender(){return gender;}publicvoidsetGender(String gender){this.gender = gender;}@OverridepublicStringtoString(){return"Emp{"+"empId="+ empId +", empName='"+ empName +'\''+", age="+ age +", gender='"+ gender +'\''+'}';}}

1. < if > 元素

在MyBatis中,< if > 是最常用的判断语句,它类似于Java中的if语句,主要用于实现某些简单的条件选择

<selectid="getEmpByyCondition"resultType="com.atguigu.mybatis.pojo.Emp">
        select * from t_emp where 1=1
        <iftest="empName!=null and empName!=''">
            and emp_name=#{empName}
        </if><iftest="age!=null and age!=''">
            and age=#{age}
        </if><iftest="gender!=null and gender!=''">
            and gender=#{gender}
        </if></select>

测试方法:

publicvoidtest(){SqlSessionUtils sqlSessionUtils =newSqlSessionUtils();SqlSession sqlSession = sqlSessionUtils.getSqlSession();EmpMapper mapper = sqlSession.getMapper(EmpMapper.class);Emp emp =newEmp(null,"张三",null,null);List<Emp> list = mapper.getEmpByyCondition(emp);System.out.println(list);}

测试结果如下:

在这里插入图片描述

同时在为传递任何参数时,程序会将数据表中的所有数据查出

2. < where >

如上述使用if语句查询我们注意到

select * from t_emp where 1=1

在语句中要添加

where 1=1

这是为了保证至少有条件成立,不至于程序报错,但是MyBatis的开发者设计了更好的方法,就是把< where >也作为元素,动态添加where,使用如下

<selectid="getEmpByConditionTwo"resultType="Emp">
        select * from t_emp
        <where><iftest="empName != null and empName != ''">
                emp_name = #{empName}
            </if><iftest="age != null and age != ''">
                and age = #{age}
            </if><iftest="gender != null and gender != ''">
                and gender = #{gender}
            </if></where></select>

这样就避免了那个设定

3. < choose >,< when >,< otherwise >元素

在使用< if >元素时,只要test属性中的表达式为true,就会执行元素中的条件语句,但是在实际应用中,有时只需要从多个选项中选择一个去执行。在这种场景下,使用< if > 元素进行处理是非常不合理的。如果使用的是Java语言,这种情况显然更适合switch语句来处理,那么MyBatis中有没有类似的语句呢?当然是有的。针对上面的情况,MyBatsi可以用< choose >,< when >,< otherwise >元素组合去实现上面的情况。

<selectid="getEmpByChoose"resultType="Emp">
        select * from t_emp
        <where><choose><whentest="empName != null and empName != ''">
                    emp_name = #{empName}
                </when><whentest="age != null and age != ''">
                    age = #{age}
                </when><whentest="gender != null and gender != ''">
                    gender = #{gender}
                </when></choose></where></select>

测试语句:

publicvoidtestGetEmpByChoose(){SqlSession sqlSession =SqlSessionUtil.getSqlSession();DynamicSQLMapper mapper = sqlSession.getMapper(DynamicSQLMapper.class);Emp emp =newEmp(null,"张三",21,"");List<Emp> list = mapper.getEmpByChoose(emp);
        list.forEach(System.out::println);}

结果如下:
在这里插入图片描述

这是因为当条件满足其中一项,在这个案例中就是满足了名字,那么后面的就不会再去匹配啦,所以这里的查询是按照名字也就是张三来进行的。

4. < trim >元素

trim元素用于自定义拼接SQL语句,与where类似,具体我们可以对比来看

< where >版

<selectid="getEmpByConditionTwo"resultType="Emp">
        select * from t_emp
        <where><iftest="empName != null and empName != ''">
                emp_name = #{empName}
            </if><iftest="age != null and age != ''">
                and age = #{age}
            </if><iftest="gender != null and gender != ''">
                and gender = #{gender}
            </if></where></select>

< trim >版

<selectid="getEmpByCondition"resultType="Emp">
        select <includerefid="empColumns"></include> from t_emp
        <trimprefix="where"suffixOverrides="and"><iftest="empName != null and empName != ''">
                emp_name = #{empName} and
            </if><iftest="age != null and age != ''">
                age = #{age} and
            </if><iftest="gender != null and gender != ''">
                gender = #{gender}
            </if></trim></select>

trim用于去掉或添加标签中的内容

常用属性:

  • prefix:在trim标签中的内容的前面添加某些内容
  • prefixOverrides:在trim标签中的内容的前面去掉某些内容
  • suffix:在trim标签中的内容的后面添加某些内容
  • suffixOverrides:在trim标签中的内容的后面去掉某些内容

5. < set >元素

在更新表中字段时,根据条件进行判断,从而实现部分更改,而不是更新所有字段,提高开发效率

<updateid="updateColumns"parameterType="Emp">
        update t_emp 
        <set><iftest="empName != null and empName != ''">
                emp_name = #{empName} and
            </if><iftest="age != null and age != ''">
                age = #{age} and
            </if><iftest="gender != null and gender != ''">
                gender = #{gender}
            </if></set></update>

其中的< if >元素用于判断相应的字段是否传入值,如果传入的字段非空,就将此字段进行动态SQL组转;并更新此字段,否则此字段不更新。

注意:在映射文件中使用 < set > 和 < if >元素组合纪念性update语句动态SQL组转时,如果< set >元素内包含的内容都为空,则会出现SQL语法错误。所以在使用< set > 元素进行字段信息更新时,要确保传入的更新字段都不能为空。

6. < foreach >元素

用于SQL语句执行批量操作

  • collection: 需做foreach(遍历)的对象,作为入参时,list、array对象时,collection属性值分别默认用"list"、"array"代替,Map对象没有默认的属性值。但是,在作为入参时可以使用@Param(“keyName”)注解来设置自定义collection属性值,设置keyName后,list、array会失效;
  • item: 集合元素迭代时的别名称,该参数为必选项;
  • index: 在list、array中,index为元素的序号索引。但是在Map中,index为遍历元素的key值,该参数为可选项;
  • open: 遍历集合时的开始符号,通常与close=")"搭配使用。使用场景IN(),values()时,该参数为可选项;
  • separator: 元素之间的分隔符,类比在IN()的时候,separator=“,”,最终所有遍历的元素将会以设定的(,)逗号符号隔开,该参数为可选项;
  • close: 遍历集合时的结束符号,通常与open="("搭配使用,该参数为可选项;

6.1 添加批量数据

<insertid="insertEmps">
        insert into t_emp value
        <foreachcollection="emps"item="emp"separator=",">
          (null,#{emp.empName},#{emp.age},#{emp.gender},null)
        </foreach></insert>

测试方法:

publicvoidtest(){SqlSessionUtils sqlSessionUtils =newSqlSessionUtils();SqlSession sqlSession = sqlSessionUtils.getSqlSession();EmpMapper mapper = sqlSession.getMapper(EmpMapper.class);Emp emp1 =newEmp(null,"数据1",21,"女");Emp emp2 =newEmp(null,"数据2",24,"男");Emp emp3 =newEmp(null,"数据3",31,"女");List<Emp> emps =Arrays.asList(emp1, emp2, emp3);
        mapper.insertEmps(emps);}

执行结果:

在这里插入图片描述

6.2 批量删除数据

<deleteid="deleteByEmpIds">
        delete from t_emp where emp_id in
        <foreachcollection="empIds"item="empId"separator=","open="("close=")">
        #{empId}
        </foreach></delete>

测试方法:

publicvoidtest(){SqlSessionUtils sqlSessionUtils =newSqlSessionUtils();SqlSession sqlSession = sqlSessionUtils.getSqlSession();EmpMapper mapper = sqlSession.getMapper(EmpMapper.class);Integer empIds[]={6,7};
        mapper.deleteByEmpIds(empIds);}

执行结果:

在这里插入图片描述

也可以这么拼接SQL语句

<deleteid="deleteByEmpIds">
        delete from t_emp where
        <foreachcollection="empIds"item="empId"separator="or">
            emp_id=#{empId}
        </foreach></delete>

7. < SQL >元素

sql片段,可以记录一段公共sql片段,在使用的地方通过include标签进行引用

<sqlid="empColumns"> emp_id,emp_name,age,gender</sql><selectid="selectAll"resultType="com.atguigu.mybatis.pojo.Emp">
        select <includerefid="empColumns"></include> from t_emp
    </select>

测试方法:

publicvoidtest(){SqlSessionUtils sqlSessionUtils =newSqlSessionUtils();SqlSession sqlSession = sqlSessionUtils.getSqlSession();EmpMapper mapper = sqlSession.getMapper(EmpMapper.class);List<Emp> emps = mapper.selectAll();System.out.println(emps);}

执行结果:

在这里插入图片描述

8. 小结

动态SQL可以帮开发者灵巧的实现很多特殊条件的SQL语句,比如if元素是最常用的,同时要灵活使用sql语句来完成嵌套查询,要根据项目要求选取合适的元素来实现开发。

标签: mybatis sql java

本文转载自: https://blog.csdn.net/m0_64102491/article/details/127477746
版权归原作者 努力的阿波 所有, 如有侵权,请联系我们删除。

“【MyBatis框架】动态SQL”的评论:

还没有评论