mybatis —— 动态sql之if条件判断各类使用方式

点滴记载,点滴进步,愿本身更上一层楼。html

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

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

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

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

maven配置。apache

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

 

非maven项目只须要将jar添加到项目中便可。mybatis

log4j.properties添加到source根目录。app

# 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
maven

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。

相关文章
相关标签/搜索