zoukankan      html  css  js  c++  java
  • 深入浅出Mybatis系列(九)---强大的动态SQL

     

      传统的使用JDBC的方法,相信大家在组合复杂的的SQL语句的时候,需要去拼接,稍不注意哪怕少了个空格,都会导致错误。Mybatis的动态SQL功能正是为了解决这种问题, 其通过 if, choose, when, otherwise, trim, where, set, foreach标签,可组合成非常灵活的SQL语句,从而提高开发人员的效率。下面就去感受Mybatis动态SQL的魅力吧:

    1. if:    你们能判断,我也能判断!

    作为程序猿,谁不懂 if !  在mybatis中也能用 if 啦:

    复制代码
    <select id="findUserById" resultType="user">
               select * from user where 
               <if test="id != null">
                   id=#{id}
               </if>
                and deleteFlag=0;
    </select>
    复制代码

    上面例子: 如果传入的id 不为空, 那么才会SQL才拼接id = #{id}。 这个相信大家看一样就能明白,不多说。

    细心的人会发现一个问题:“你这不对啊! 要是你传入的id为null,  那么你这最终的SQL语句不就成了 select * from user where and deleteFlag=0,  这语句有问题!”

    是啊,这时候,mybatis的 where 标签就该隆重登场啦:

    2. where, 有了我,SQL语句拼接条件神马的都是浮云!

    咱们通过where改造一下上面的例子:

    复制代码
    <select id="findUserById" resultType="user">
               select * from user 
    <where> <if test="id != null"> id=#{id} </if> and deleteFlag=0; </where> </select>
    复制代码

    有些人就要问了: “你这都是些什么玩意儿! 跟上面的相比, 不就是多了个where标签嘛! 那这个还会不会出现  select * from user where and deleteFlag=0 ?”

    的确,从表面上来看,就是多了个where标签而已, 不过实质上, mybatis是对它做了处理,当它遇到AND或者OR这些,它知道怎么处理。其实我们可以通过 trim 标签去自定义这种处理规则。

    3. trim :  我的地盘,我做主!

    上面的where标签,其实用trim 可以表示如下:

    <trim prefix="WHERE" prefixOverrides="AND |OR ">
      ... 
    </trim>

    它的意思就是: 当WHERE后紧随AND或则OR的时候,就去除AND或者OR。 除了WHERE以外, 其实还有一个比较经典的实现,那就是SET。

    4. set:  信我,不出错!

    复制代码
    <update id="updateUser" parameterType="com.dy.entity.User">
               update user set 
               <if test="name != null">
                   name = #{name},
               </if> 
               <if test="password != null">
                   password = #{password},
               </if> 
               <if test="age != null">
                   age = #{age}
               </if> 
               <where>
                   <if test="id != null">
                       id = #{id}
                   </if>
                   and deleteFlag = 0;
               </where>
    </update>
    复制代码

    问题又来了: “如果我只有name不为null,  那么这SQL不就成了 update set name = #{name}, where ........ ?  你那name后面那逗号会导致出错啊!”

    是的,这时候,就可以用mybatis为我们提供的set 标签了。下面是通过set标签改造后:

    复制代码
    <update id="updateUser" parameterType="com.dy.entity.User">
               update user
    <set> <if test="name != null">name = #{name},</if> <if test="password != null">password = #{password},</if> <if test="age != null">age = #{age},</if> </set> <where> <if test="id != null"> id = #{id} </if> and deleteFlag = 0; </where> </update>
    复制代码

    这个用trim 可表示为:

    <trim prefix="SET" suffixOverrides=",">
      ...
    </trim>

    WHERE是使用的 prefixOverrides(前缀), SET是使用的 suffixOverrides (后缀), 看明白了吧!

    5. foreach:  你有for, 我有foreach, 不要以为就你才屌!

    java中有for, 可通过for循环, 同样, mybatis中有foreach, 可通过它实现循环,循环的对象当然主要是java容器和数组。

    复制代码
    <select id="selectPostIn" resultType="domain.blog.Post">
      SELECT *
      FROM POST P
      WHERE ID in
      <foreach item="item" index="index" collection="list"
          open="(" separator="," close=")">
            #{item}
      </foreach>
    </select>
    复制代码

    将一个 List 实例或者数组作为参数对象传给 MyBatis,当这么做的时候,MyBatis 会自动将它包装在一个 Map 中并以名称为键。List 实例将会以“list”作为键,而数组实例的键将是“array”。同样, 当循环的对象为map的时候,index其实就是map的key。

    6. choose:  我选择了你,你选择了我!

    Java中有switch,  mybatis有choose。

    复制代码
    <select id="findActiveBlogLike"
         resultType="Blog">
      SELECT * FROM BLOG WHERE state = ‘ACTIVE’
      <choose>
        <when test="title != null">
          AND title like #{title}
        </when>
        <when test="author != null and author.name != null">
          AND author_name like #{author.name}
        </when>
        <otherwise>
          AND featured = 1
        </otherwise>
      </choose>
    </select>
    复制代码

    以上例子中: 当title和author都不为null的时候, 那么选择二选一(前者优先), 如果都为null, 那么就选择 otherwise中的, 如果tilte和author只有一个不为null, 那么就选择不为null的那个。

    纵观mybatis的动态SQL, 强大而简单, 相信大家简单看一下就能使用了。

    好啦,本次就写到这!下篇文章将结合mybatis的源码分析一次sql语句执行的整个过程。

  • 相关阅读:
    关于celery踩坑
    关于git的分批提交pull requests流程
    SymGAN—Exploiting Images for Video Recognition: Heterogeneous Feature Augmentation via Symmetric Adversarial Learning学习笔记
    AFN—Larger Norm More Transferable: An Adaptive Feature Norm Approach for Unsupervised Domain Adaptation学习笔记
    Learning to Transfer Examples for Partial Domain Adaptation学习笔记
    Partial Adversarial Domain Adaptation学习笔记
    Partial Transfer Learning with Selective Adversarial Networks学习笔记
    Importance Weighted Adversarial Nets for Partial Domain Adaptation学习笔记
    Exploiting Images for Video Recognition with Hierarchical Generative Adversarial Networks学习笔记
    improved open set domain adaptation with backpropagation 学习笔记
  • 原文地址:https://www.cnblogs.com/a8457013/p/7830825.html
Copyright © 2011-2022 走看看