zoukankan      html  css  js  c++  java
  • mybatis —— 动态sql之if条件判断各种使用方式

    点滴记载,点滴进步,愿自己更上一层楼。

    用mybatis执行数据库操作仅仅能看到执行结果,如果想看到执行的sql语句怎么办。

    查阅mybatis官方文档找到了解决方法。

    配置什么的很简单,用的log4j打印,当然参照官方文档还有好几种方法,具体自弄。

    这里仅作记录只用。配置很简单,将log4j架包加入到classpath里。

    maven配置。

            <dependency>
                <groupId>log4j</groupId>
                <artifactId>log4j</artifactId>
                <version>1.2.17</version>
            </dependency>

    非maven项目只需要将jar添加到项目中即可。

    log4j.properties添加到source根目录。

    # Global logging configuration
    log4j.rootLogger=ERROR, stdout
    # MyBatis logging configuration...
    #log4j.logger.com.soft.test.dao=DEBUG
    log4j.logger.dynamic=DEBUG
    #log4j.logger.org.mybatis.example.BlogMapper=TRACE
    # Console output...
    log4j.appender.stdout=org.apache.log4j.ConsoleAppender
    log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
    log4j.appender.stdout.layout.ConversionPattern=%5p [%t] - %m%n

    其中关键的地方是
    log4j.logger.dynamic=DEBUG

    log4j.logger是固定的,dynamic为你的mapper.xml的namespace

    如果我的xml中的namespace为dynamic

    <?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">
    <!-- namespace命名空间,跟java的package类似,避免sql id重复,
    有了这个命名空间,别的xml中的sql的id可以跟这个重复,并且 namespace不能省略,不能为空,不用接口开发,此处可以随意写-->
    <mapper namespace="dynamic">
     
        <resultMap id="userMap" type="com.soft.test.model.User">
            <id column="id" property="id"/>
            <result column="username" property="username"/>
            <result column="password" property="password"/>
            <result column="create_date" property="createDate"/>
        </resultMap>
     
        <!--if 标签使用类似html的C标签的if -->
        <select id="selectUseIf" parameterType="com.soft.test.model.User" resultMap="userMap">
            select * from t_user where
            <if test="id != null and id != ''">
                id=#{id}
            </if>
            <if test="username != null and username != ''">
                and username like concat('%',#{username},'%')
            </if>
            <if test="password != null and password != ''">
                and password=#{password}
            </if>
        </select>
        
    </mapper>

    配置完成。现在运行测试即可看到运行的sql语句
    ------------------------------------------------------------------------------------------------------------------------------

    DEBUG [main] - ==>  Preparing: select * from t_user where id=? and username like concat('%',?,'%') 
    DEBUG [main] - ==> Parameters: 28(Integer), xiao(String)
    DEBUG [main] - <==      Total: 1

    -------------------------------------------------------------------------------------------------------------------------------

    mybatis的if判断语句其实跟el表达式的if条件判断有些类似。

    例如: <if test="id != null"> </if>

    1 如果参数为数字类型的时候没有特俗需求的情况只需要判断是否为null即可。

    例如:<if test="id != null"></if>

     如果有特俗需求,例如判断是否大于某个数的时候才行。只需要加上对应的条件判断即可

    例如:<if test='id != null and id > 28'></if>

     mybatis对于这种大于小于等等还有另一种形式。

    例如:<if test='id != null and id gt 28'></if>

    对应关系:

    ---------------------------------------

        gt            对应             >

        gte         对应              >=

        lt             对应              <(会报错  相关联的 "test" 属性值不能包含 '<' 字符)

        lte          对应               <=(会报错  相关联的 "test" 属性值不能包含 '<' 字符)

    ---------------------------------------

    2 如果为字符串类型

    2.1 如果不需要过滤空串的情况 仅仅判断null即可

    例如:<if test="username != null"></if>

    2.2 如果需要过滤空串,添加空串判断即可  不支持 &&   所以这里用 and  or  || 来做逻辑与或的判断 

    例如:<if test="username != null and '' != username"></if> 或者 <if test="username != null and ''  neq username"></if>

    2.3 如果判断字符串是否已某个特俗字符开头,结尾等。直接调用String的对应方法即可

    例如:<if test="username != null and username.indexOf('ji') == 0"> </if> <!-- 是否以什么开头 -->
        <if test="username != null and username.indexOf('ji') >= 0"> </if> <!-- 是否包含某字符 -->
        <if test="username != null and username.lastIndexOf('ji') > 0"></if>  <!-- 是否以什么结尾 -->

    2.4 是否是某个特定字符串,某些业务有此需要。

    例如:<if test="username != null and 'hello' == username"></if> 或者<if test="username != null and 'hello' eq username"></if>

    注意:

    <if test="username != null and 'hello' == username"></if>这种形式的写法在参数类型是字符串的时候是没有问题的,

    但是参数类型为非字符串类型的时候就需要写成 <if test="username != null and 'hello'.toString() == username.toString()"></if>

    仅仅写成<if test="username != null and 'hello'.toString() == username"></if>也会有很大可能会挂。

    也许你会说非字符串的为什么要写成这样。这就要看特俗需要了。

    例如:某一个sql片段是公用的,

    <if test="username != null"></if>
    <if test="password != null"></if>

    该片段更新条件也用,但是当你需要将某一个字段更新成null的时候怎么办。

    这个时候就可以通过传入一个特定的字符串来弄。当传入的字符串为特定字符串的时候就更新该字符串为null。

    <if test="username != null and 'hello'.toString() == username.toString()">xxx=null</if>

    当然这样子貌似date型会挂。

    通过 2.2 也可以看出mybatis对于字符串的相等不相等的判断也是有对应的特俗操作符的。

    -------------------------------------------------------

    eq                  对应                ==

    neq               对应                 !=

    ------------------------------------------------------

    当然还可以看出来if的条件判断test是支持对象自身方法调用的,即使是自己写的方法,可以自己尝试。当然下面会有例子。

    例如:里面可以用‘xxxx’.equals(xxxx) 字符串的比较两个字符串方法

        xxxx.indexOf('ss') 判断字符串里面是否包含某个字符等等  

     3 判断list是否为空

    上面说过,if条件判断可以直接调用对象自身的方法进行逻辑判断,所以list判空。可以调用.size()>0或者.isEmpty()

    例如:<if test="userList != null and userList.isEmpty()"></if> , <if test="userList != null and userList.size()>0"></if>

    4 map参数同同理  取值的话 map.key(map中的key名字)即可

    ----------------------------------------------------------------------------------------------分割线01-----------------------------------------------------------------------------------------------------------

    这里是上面的各种理论的实践。这里可以不看,自己去实践最好。

    1 数字类型。

    仅作null判断。

     <!--if 标签使用类似html的C标签的if -->
        <select id="selectUseIf" parameterType="com.soft.test.model.DynamicTestModel" resultMap="userMap">
            select * from t_user where 1=1
            <if test='id != null'>
                and id=#{id}
            </if>
        </select>

    当id不为null的时候 打印的log
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and id=? 
    DEBUG [main] - ==> Parameters: 28(Integer)
    DEBUG [main] - <==      Total: 1

    当id为null的时候 打印的log
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 
    DEBUG [main] - <==      Total: 1

    两项对比,可以看出id=?这个条件随着传入参数id的变化而变化。

    当有特俗需求的时候,当前数据库中的表id为28 仅仅有这么一条数据,做一下轻微的改动

    <if test='id != null and id > 27 '>

    当传入id=28的时候
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and id=? 
    DEBUG [main] - ==> Parameters: 28(Integer)
    DEBUG [main] - <==      Total: 1

    当传入id小于28的时候

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 
    DEBUG [main] - <==      Total: 1

    接下来测试下面这几种对应关系。

    ---------------------------------------

        gt            对应             >

        gte         对应              >=


        lt             对应              <(会报错  相关联的 "test" 属性值不能包含 '<' 字符)

                lte          对应               <=(会报错  相关联的 "test" 属性值不能包含 '<' 字符)

    ---------------------------------------

    gt   

    <if test='id != null and id gt 27 '>

    参数  id=25
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters:

    参数  id=28
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and id=? 
    DEBUG [main] - ==> Parameters: 28(Integer)

     >= 

    <if test='id != null and id >= 28 '>

    参数  id=28
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and id=? 
    DEBUG [main] - ==> Parameters: 28(Integer)

    参数  id=27

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 

    gte   

    <if test='id != null and id gte 28 '>

    参数  id=28
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and id=? 
    DEBUG [main] - ==> Parameters: 28(Integer)

    参数  id=27

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 

        使用 <   <= 直接报错       "test" 属性值不能包含 '<' 字符 看来只能用 lt   lte了

    lt

    <if test='id != null and id lt 28 '>

    参数  id=28  
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 
    参数  id=27

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and id=? 
    DEBUG [main] - ==> Parameters: 27(Integer)

    lte

    <if test='id != null and id lte 28 '>

    参数  id=28  

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and id=? 
    DEBUG [main] - ==> Parameters: 28(Integer)

    参数  id=29

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 

    ----------------------------------------------------------------------------------------------分割线02-----------------------------------------------------------------------------------------------------------

    2.1 跟1的第一条一样不做重复测试

    2.2 过滤空串

    <select id="selectUseIf" parameterType="com.soft.test.model.DynamicTestModel" resultMap="userMap">
            select * from t_user where 1=1
            <if test="username != null and '' != username ">
                and username=#{username}
            </if>
        </select>

    !=
    username=“xiao”
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiao(String)

    username=“”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 

    neq

    <if test="username != null and '' neq username ">

    username=“xiao”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiao(String)

    username=“”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 

    各个逻辑与或的判断
    and上面已经弄过了,这里弄or  || 两种条件  

    <if test="'xiaohong' eq username or 'xiao' eq username ">

    or

    username=“xiao”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiao(String)
    username=“xiaohong”
    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiaohong(String)

    username=“xiaofang”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 

    ||

    username=“xiao”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiao(String)


    username=“xiaohong”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiaohong(String)

    username=“xiaofang”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 
    DEBUG [main] - ==> Parameters: 

    2.3  indexOf()  lastIndexOf()  判断是否包含某个特定字符

    <if test="username != null and username.indexOf('xiao')==0">

    username=“xiao”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiao(String)

    username=“xiaofang”

    DEBUG [main] - ==>  Preparing: select * from t_user where 1=1 and username=? 
    DEBUG [main] - ==> Parameters: xiaofang(String)
    其他两个没什么大不同的。自行测试。
    2.4   判断是否是某个字符

    <if test="'xiaohong' eq username">

    username=“xiaohong”
    ==>  Preparing: select * from t_user where 1=1 and username=? 
    ==> Parameters: xiaohong(String)

    username=“xiaofang”

    ==>  Preparing: select * from t_user where 1=1 
    ==> Parameters: 

    3  4  的本质就是再说mybatis的if条件判断语句可以直接执行对象的方法。下面自己写一个方法,在if里面试试。

    自己定义一个类,里面一个方法用于条件判断用。

    public class DynamicSql1Model {
        public boolean getMySelfMethod(){
            return true;
        }
    }

    该类作为一个属性放入到model中  仅仅贴出部分代码

    public class DynamicSqlModel {
        private int id;
        private String username;
        private String password;
        private Date createDate;
        private List<String> list;
        private Map<String,Object> mapParam;
     
        private DynamicSql1Model dynamicSql1Model;

    xml中引用该model的方法

     <if test="dynamicSql1Model.getMySelfMethod()">

    开始测试

    DynamicSqlModel user = new DynamicSqlModel();
            user.setUsername("xiaofang");
            user.setPassword("123456");
            user.setCreateDate(new Date());
            DynamicSql1Model dynamicSqlModel = new DynamicSql1Model();
            user.setDynamicSql1Model(dynamicSqlModel);
            dao.selectUseIf(user);

    现在返回结果
    现在方法的返回值为true 
    ==>  Preparing: select * from t_user where 1=1 and username=? 
    ==> Parameters: xiaofang(String)
    方法返回值修改为false

    ==>  Preparing: select * from t_user where 1=1 
    ==> Parameters: 
    可以看出完全可以使用自定义的方法进行if条件控制。通过该特性可以干一些特俗业务的事情。自己体会。

    本篇说的主要是if条件判断动态控制sql。可以看出有弊端。因为if条件不满足的时候sql会变成

    select * from t_user where  所以我在条件后面加了个 1=1 但是这是不符合逻辑的。下节介绍where以及其他标签用于动态sql。

  • 相关阅读:
    fiber
    ACM用到的算法。先做个笔记,记一下
    matlab安装及破解
    银行家算法
    网络安全(超级详细)零基础带你一步一步走进缓冲区溢出漏洞和shellcode编写!
    心脏滴血漏洞复现(CVE-2014-0160)
    KMP算法分析
    利用BURPSUITE检测CSRF漏洞
    BURPSUITE爆破密码
    动态规划—最长回文子串LEETCODE第5题深度剖析
  • 原文地址:https://www.cnblogs.com/sumlen/p/11130554.html
Copyright © 2011-2022 走看看