聊聊mybatis的动态sql

说在前面

最近一直忙着做项目,都没什么时间写写博客,真是不开心。
今天想聊聊Mybatis的动态sql,采用的还是老形式,通过源码分析:


MyBatis中用于实现动态SQL的元素主要有:

  • if
  • choose
  • where
  • trim
  • set
  • foreach

我们先从简单的 if 说起:

<select id="getTotalRows" resultType="java.lang.Integer"
    parameterType="java.lang.String">
    SELECT * FROM mvc_user_info u where u.user_name = 'lixifan'
        <if test="trueName != null and trueName != ''">
            AND u.true_name LIKE CONCAT('%',#{trueName},'%')
        </if>
        <if test="mail != null and mail != ''">
            AND u.mail LIKE CONCAT('%',#{mail},'%')
        </if>
</select>
这几行代码的意思其实就是如果你传递了trueName,那么就要满足条件trueName = #{trueName},如果传递了mail,那么就要满足mail = #{mail},这种情况可以用于两种应用场景即只传trueName查出数据和只传mail 查出数据。我在开发中有多次使用这种情况,甚至有一次性将整个对象的数据成员都用if进行判断,通过这种方式来提高开发的速率。

看完上面那个if的用法,很多人应该会想如何轻易做到只需满足一种条件即可的,比如java的switch,别急,现在聊聊choose咯:

<select id="chooseTest" parameterType="Blog" resultType="Blog">  
select * from mvc_user_info u where u.user_name = 'lixifan'
<choose>  
    <when test="trueName != null">  
        and u.true_name = #{trueName }  
    </when>  
    <when test="mail != null">  
        and u.mail = #{mail }  
    </when>  
    <otherwise>  
        and age= "22"  
    </otherwise>  
</choose>  

</select>
看到了吗,其实choose元素相当于java的switch语句,而when其实就相当于case,otherwise就是default,具体用法也是跟switch一样的啦,我就不多说了!


嗯,现在看看where

<select id="getTotalRows" resultType="java.lang.Integer"
    parameterType="java.lang.String">
    SELECT count(*) FROM mvc_user_info u LEFT JOIN mvc_dept_info d ON
    u.dept_id = d.dept_id
    <where>
        <if test="trueName != null and trueName != ''">
            AND u.true_name LIKE CONCAT('%',#{trueName},'%')
        </if>
        <if test="mail != null and mail != ''">
            AND u.mail LIKE CONCAT('%',#{mail},'%')
        </if>
        <if test="userLevel != null and userLevel != ''">
            AND u.user_level = #{userLevel}
        </if>
        <if test="deptNo != null and deptNo != ''">
            AND d.dept_no LIKE CONCAT('',#{deptNo},'%')
        </if>
    </where>
</select>

where元素的作用就是在写入where元素的地方插入一个where,而最大的功效应该是:(智能处理)

  • 如果所有的条件都不满足那么MyBatis就会查出所有的记录
  • 如果输出后是and 开头的,MyBatis会把第一个and忽略
  • 在where元素中你不需要考虑空格的问题,MyBatis会智能的帮你加上

聊聊trim

<insert id="saveUser" parameterType="cn.springmvc.model.UserInfo">
    INSERT INTO mvc_user_info
    <trim prefix="(" suffix=")" suffixOverrides=",">
        <if test="userName != null">
            user_name,
        </if>
        <if test="trueName != null">
            true_name,
        </if>
    </trim>
    VALUES
    <trim prefix="(" suffix=")" suffixOverrides=",">
        <if test="userName != null">
            #{userName, jdbcType=VARCHAR},
        </if>
        <if test="trueName != null">
            #{trueName, jdbcType=VARCHAR},
        </if>
    </trim>
</insert>

先说说几个属性:
prefix:加上前缀
suffix:加上后缀
suffixOverrides:将尾部的指定内容覆盖掉
prefixOverrides:将首部的指定内容覆盖掉
而结合上面的源码来说就是:
构建成 INSERT INTO mvc_user_info(xx,yy) values(aa,bb);


现在聊聊set

<update id="test" parameterType="Blog">  
update t_blog  
<set>  
    <if test="title != null">  
        title = #{title},  
    </if>  
    <if test="content != null">  
        content = #{content},  
    </if>  
    <if test="owner != null">  
        owner = #{owner}  
    </if>  
</set>  
where id = #{id}  

</update>
set元素主要是用在更新操作的时候,它的主要功能和where元素其实是差不多的,主要是在包含的语句前输出一个set,然后如果包含的语句是以逗号结束的话将会把该逗号忽略,如果set包含的内容为空的话则会出错。有了set元素我们就可以动态的更新那些修改了的字段。


最后一个foreach了哈

foreach的主要用在构建in条件中,它可以在SQL语句中进行迭代一个集合。
foreach元素的属性主要有item,index,collection,open,separator,close。
item表示集合中每一个元素进行迭代时的别名;
index指定一个名字,用于表示在迭代过程中,每次迭代到的位置;
open表示该语句以什么开始;
separator表示在每次进行迭代之间以什么符号作为分隔符;
close表示以什么结束;
实例如下:

<delete id="deleteUser" parameterType="String">
    DELETE FROM mvc_user_info WHERE user_id IN
    <foreach collection="array" item="idItem" open="(" separator=","
        close=")">
        #{idItem}
    </foreach>
</delete>

Note:发布的这些文章全都是自己边学边总结的,难免有纰漏,如果发现有不足的地方,希望可以指出来,一起学习咯,么么哒。
开源爱好者,相信开源的力量必将改变世界:
github: https://github.com/wiatingpub
  osc  : https://git.oschina.net/xi_fan

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容

  • 1. 简介 1.1 什么是 MyBatis ? MyBatis 是支持定制化 SQL、存储过程以及高级映射的优秀的...
    笨鸟慢飞阅读 5,422评论 0 4
  • MyBatis 的强大特性之一便是它的动态 SQL。如果你有使用 JDBC 或其他类似框架的经验,你就能体会到根据...
    jackcooper阅读 486评论 0 1
  • 1 动态SQL# 那么,问题来了: 什么是动态SQL? 动态SQL有什么作用? 传统的使用JDBC的方法,相信大家...
    七寸知架构阅读 18,622评论 2 58
  • 十五年前的今天,你来到这个世界,0:37分,这个值得纪念的时刻,这个令人兴奋的日子,你就那样准时到来,与医生说的...
    娜木翰阅读 1,193评论 11 5
  • 秋风扫落叶,落叶铺大地。 尽在天空看悠悠,流沦的时间挥挥而旧。 依旧依旧,潇洒身后。 常春藤稀稀疏疏的遮掩着那些淡...
    啊梓阅读 273评论 0 1