百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分类 > 正文

mybatis动态SQL常用语法总结(mybatis动态sql常用语法总结汇总)

ztj100 2024-11-02 14:29 23 浏览 0 评论

在 mybatis 的 xml 文件里写的 sql 语句实际用的是一门叫做 OGNL 的表达式语言,OGNL 全称 Object Graph Navigation Language 对象图导航语言,是常应用于 Java 中的一个开源的表达式语言(Expression Language),它被集成在 Spring、Mybatis、Struts2 等 Java 框架中,通过简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图,实现类型转化等功能。

在书写动态 SQL 时经常需要借助各种标签,下面是一些在 mybatis 中常用的标签:

if 标签

  • 没有 else、else if 标签
  • if 标签可以嵌套
  • if 标签判断条件 test 中可以用 or 的,我们经常看到用 and
  • 对于 number 类型,0 或者浮点型 0.00 会被当成 false(和 js 类似)
  • 对于字符串类型才需要判断 != '',其他类型直接判断 != null 就行了
  • 对于空字符串 "" 会被当成 false
  • 单引号内只有一个字符时,OGNL 会识别成 java 中的 char 类型,然后数据如果是 String 类型时会导致判断失效,可以直接将 test 的引号改成单引号,然后里面的字符串用双引号

if 标签的 test 中常用判断:

* 相等:==
* 不等:!=
* 条件或:or
* 条件与:and
* 条件非:!,也可以用 not
* 包含:in
* 不包含:not in
* 小于:<
* 小于等于:<=
* 大于:>
* 大于等于:>=

choose、when、otherwise 标签

相当于 if、else if、else,间接实现了上面 if 标签不支持的 else 效果

where 标签

用于拼接 SQL 语句中的 where 子句,条件成立时才会加上 where 关键字,可以避免拼接出多余的and、or

set 标签

  • 用于拼接SQL语句中的set子句,与 update 语句配合使用,条件成立时才会加上 set 关键字,可以避免拼接出多余的逗号,如果有多余的标签 set 标签会自动去掉
  • 一定会加上 set 关键字,所以要保证至少有一个条件成立,否则会报 SQL 语法错误
<update id="updateUserInfo" parameterType="UserInfo">
    update t_user_info
    <set>
        <if test="userName != null and userName != ''">
            user_name = #{userName},
        </if>
        <if test="age != null">
            age = #{age},
        </if>
    </set>
    where user_id = #{userId}
</update>

trim 标签

可以间接实现 where 和 set 标签一样的功能

foreach 标签

遍历集合类数据,标签属性:

  • collection:要被遍历解析的对象,集合名或者数组名
  • item:集合或数组中每一个迭代元素的别名
  • index:在 list 和数组中为元素序号,在 map 中为元素的 key
  • open:开始符号
  • close:结束符号
  • separator:连接每一项的分割符号

collection 接收的参数:

  • 匿名参数:当在 java 方法中没有通过 @Param 注解指定参数名时,列表类型默认参数名为 list,数组为 array,注意 Map 无默认值需要自己指定具名参数
  • 具名参数:java 方法中使用了 @Param 注解指定了参数名称,则 foreach 中的 collection 属性必须为指定的参数名

示例1:匿名参数

<!-- mapper -->
int insertUsers(List<User> users);
int updateUsers(List<User> users);

<!-- xml -->
<insert id="insertUsers">
    insert into t_user (id, user_name, age) values
    <foreach collection="list" separator="," item="user">
        (#{user.user_id}, #{user.user_name}, #{user.age})
    </foreach>
</insert>

<update id="updateUsers">
    <foreach collection="list" separator=";" item="user">
        update t_user
        <set>
            <if test="user.userName != null and user.userName != ''">
                user_name= #{user.userName},
            </if>
            <if test="user.age != null">
                age= #{user.age},
            </if>
        </set>
        where user_id = #{user.userId}
    </foreach>
</update>

示例2:具名参数

<!-- mapper -->
List<User> selectUsers(@Param("userIds") String[] userIds);

<!-- xml -->
<select id="selectUsers" resultType="User">
     select * from t_user where user_id in
     <foreach collection="userIds" item="item" open="(" close=")" separator=",">
        #{item}
     </foreach>
</select>

更多参数传递可以参考后面的 mybatis 参数章节。

sql、include 标签

可以在 sql 标签里定义语句,然后在需要的地方用 include 标签引入进去,可以实现代码片段复用。

selectKey 标签

用于配合插入数据成功后返回的数据,一般用来返回 id 之类的。

mybatis 中参数传递

主要有以下 5 种传参方式:

  • 匿名参数
  • 具名参数,需用 @Param 注解
  • List、Array、Set 参数
  • Map 参数
  • Java Bean 参数
  • JSON 参数

1、匿名参数

单个简单类型参数

xml 获取的时候可以随便写,mybatis 会去自动处理,反正只有一个参数干脆就让你写啥都无所谓,不过推荐还是写个有意义的形参:

<!-- mapper -->
User getUserByUsername(String userName);

<!-- xml -->
<select id="getUserByUsername" resultType="User">
    select * from t_user where user_name = #{xxoo}
</select>

多个简单类型参数

多个匿名参数的时候只能通过 mybatis 内置的 param1、param2 按传参顺序对应:

<!-- mapper -->
List<User> selectByuserNameAndAge(String userName, Integer age);

<!-- xml -->
<select id="selectByuserNameAndAge" resultMap="BaseResultMap" >
    select *  from t_user where user_name = #{param1} and age = #{param2}
</select>

2、具名参数

具名参数需要用 @Param 注解来指定

<!-- mapper -->
List<User> selectByuserNameAndAge(@Param("name") String userName, @Param("age") Integer age);

<!-- xml -->
<select id="selectByuserNameAndAge" resultMap="BaseResultMap" >
    select *  from t_user where user_name = #{name} and age = #{age}
</select>

3、List、Array、Set 参数

List 类型参数默认值 list,Array 类型参数默认值 array,注意 Set 类型的默认值并不是 set 而是 collection,除此之外的集合默认值也是 collection,当然我们也可以用 @Param 注解来自己指定

<!-- mapper -->
List<User> selectByAgeList(List ages);

<!-- xml -->
<select id="selectByList" resultMap="BaseResultMap" >
    SELECT * from t_user where age in
    <foreach collection="list" open="(" separator="," close=")" item="age">
        #{age}
    </foreach>
</select>

4、Map 参数

使用 Map 参数时,可以直接用键名引用

Map params = new HashMap<>();
params.put("userName", "周小黑");
params.put("age", 18);
List<User> result = userMapper.selectByMapParams(params);

<!-- mapper -->
List<User> selectByMapParams(Map params);

<!-- xml -->
<select id="selectByMapParams" resultMap="BaseResultMap" parameterType="map">
  select * from t_user where user_name = #{userName} and age = #{age}
</select>

5、Java Bean 参数

和上面的 Map 比较类似,不过这里的 parameterType 要指定为对应的 Bean 实体类型:

<!-- mapper -->
List<User> selectByBeans(User user);

<!-- xml -->
<select id="selectByBeans" resultMap="BaseResultMap" parameterType="User">
    select * from t_user where user_name = #{userName} and age = #{age}
</select>

6、JSON 参数

和上面的 Map、Bean 参数类似,一般是直接把前端传递过来的 json 参数直接传入 Mapper 中进行查询,parameterType 为 JSONObject

<!-- mapper -->
List<User> selectByJSON(JSONObject params);

<!-- xml -->
<select id="selectByJSON" resultMap="BaseResultMap" parameterType="com.alibaba.fastjson.JSONObject">
    select * from t_user where user_name = #{userName} and age = #{age}
</select>

常见属性设置:

  • resultMap:当查询的结果需要进行复杂的映射,例如将结果映射到具有复杂关系的对象上时,resultMap允许自定义结果集的映射规则。如果 xml 文件内有使用 resultMap 标签定义好结果集数据,需要返回的时候直接写 resultMap="BaseResultMap" 就行了,mybatis 会智能地自动给你映射过去。
  • resultType: 当查询的结果可以简单地映射到一个 POJO 对象或基本数据类型/包装类时,使用resultType,MyBatis会尝试自动将结果集映射到resultType指定的类型的对象上。
  • parameterType: 用于指定传入SQL语句的参数类型,MyBatis会根据parameterType的类型,将方法参数自动转换为SQL语句所需的类型

常用转义符

XML 文件和我们常见的 HTML 类似,都是通过标签来定义数据,而尖括号本身就是表示标签符号的开始和结束,所以在 mybatis 的 xml 文件中相关符号最好用转义符,尤其小于符号 "<",这样可以避免解析时报错,常用转义符:

字符名称      sql符号   转义字符
大于号          >           >
小于号          <           <
大于等于号    >=        >=
小于等于号    <=        <=
单引号           '            '
双引号          "            "

常用 jdbcType类型对应的 Java 类型

jdbcType                JavaType
CHAR                     String
VARCHAR               String
LONGVARCHAR      String
NUMERIC                BigDecimal
DECIMAL                 BigDecimal
BOOLEAN                boolean
TINYINT                   byte
SMALLINT               short
INTEGER                  int
BIGINT                     long
FLOAT                     double
DOUBLE                  double
DATE                       Date
TIME                       Time
TIMESTAMP            Timestamp

相关推荐

其实TensorFlow真的很水无非就这30篇熬夜练

好的!以下是TensorFlow需要掌握的核心内容,用列表形式呈现,简洁清晰(含表情符号,<300字):1.基础概念与环境TensorFlow架构(计算图、会话->EagerE...

交叉验证和超参数调整:如何优化你的机器学习模型

准确预测Fitbit的睡眠得分在本文的前两部分中,我获取了Fitbit的睡眠数据并对其进行预处理,将这些数据分为训练集、验证集和测试集,除此之外,我还训练了三种不同的机器学习模型并比较了它们的性能。在...

机器学习交叉验证全指南:原理、类型与实战技巧

机器学习模型常常需要大量数据,但它们如何与实时新数据协同工作也同样关键。交叉验证是一种通过将数据集分成若干部分、在部分数据上训练模型、在其余数据上测试模型的方法,用来检验模型的表现。这有助于发现过拟合...

深度学习中的类别激活热图可视化

作者:ValentinaAlto编译:ronghuaiyang导读使用Keras实现图像分类中的激活热图的可视化,帮助更有针对性...

超强,必会的机器学习评估指标

大侠幸会,在下全网同名[算法金]0基础转AI上岸,多个算法赛Top[日更万日,让更多人享受智能乐趣]构建机器学习模型的关键步骤是检查其性能,这是通过使用验证指标来完成的。选择正确的验证指...

机器学习入门教程-第六课:监督学习与非监督学习

1.回顾与引入上节课我们谈到了机器学习的一些实战技巧,比如如何处理数据、选择模型以及调整参数。今天,我们将更深入地探讨机器学习的两大类:监督学习和非监督学习。2.监督学习监督学习就像是有老师的教学...

Python教程(三十八):机器学习基础

...

Python 模型部署不用愁!容器化实战,5 分钟搞定环境配置

你是不是也遇到过这种糟心事:花了好几天训练出的Python模型,在自己电脑上跑得顺顺当当,一放到服务器就各种报错。要么是Python版本不对,要么是依赖库冲突,折腾半天还是用不了。别再喊“我...

超全面讲透一个算法模型,高斯核!!

...

神经网络与传统统计方法的简单对比

传统的统计方法如...

AI 基础知识从0.1到0.2——用“房价预测”入门机器学习全流程

...

自回归滞后模型进行多变量时间序列预测

下图显示了关于不同类型葡萄酒销量的月度多元时间序列。每种葡萄酒类型都是时间序列中的一个变量。假设要预测其中一个变量。比如,sparklingwine。如何建立一个模型来进行预测呢?一种常见的方...

苹果AI策略:慢哲学——科技行业的“长期主义”试金石

苹果AI策略的深度原创分析,结合技术伦理、商业逻辑与行业博弈,揭示其“慢哲学”背后的战略智慧:一、反常之举:AI狂潮中的“逆行者”当科技巨头深陷AI军备竞赛,苹果的克制显得格格不入:功能延期:App...

时间序列预测全攻略,6大模型代码实操

如果你对数据分析感兴趣,希望学习更多的方法论,希望听听经验分享,欢迎移步宝藏公众号...

AI 基础知识从 0.4 到 0.5—— 计算机视觉之光 CNN

...

取消回复欢迎 发表评论: