为了熟练掌握动态SQL你必须要知道Mybatis中的OGNL表达式

举报
码农小胖哥 发表于 2022/03/31 23:01:30 2022/03/31
【摘要】 前言 OGNL是个什么东西?很多刚入门Java的同学会有点陌生。但是在Structs流行的时代OGNL可是必会的数据渲染技术。它全称Object Graph Navigation Language,作用是降低对数据层访问的难度,它拥有类型转换、访问对象方法、操作集合对象等功能。目前已经很少通过OGNL来访问数据层了,写这篇文...

前言

OGNL是个什么东西?很多刚入门Java的同学会有点陌生。但是在Structs流行的时代OGNL可是必会的数据渲染技术。它全称Object Graph Navigation Language,作用是降低对数据层访问的难度,它拥有类型转换、访问对象方法、操作集合对象等功能。目前已经很少通过OGNL来访问数据层了,写这篇文章主要是因为目前国内大部分的ORM框架是Mybatis,而Mybatis中的动态SQL技术运用了OGNL

Mybatis中的OGNL

不少人在Mybatis的Mapper文件中写过这样的判断:


   
  1. <if test="field!='' and field!= null">
  2.   and some_col = #{field}
  3. </if>

field不为空字符并且不为null的情况下增加一个查询条件。其中 test就是一个OGNL表达式。Mybatis中的OGNL表达式主要有两种用途。

条件断言

这种是我们最常用的。执行动态SQL的条件断言,常用的有这些表达式:

  • b1 or b2  条件

  • b1 and b2  条件

  • !b1 取反,也可以写作not b1

  • b1 == b2,b1 eq b2  判断两个值相等

  • b1 != b2,b1 neq b2 判断两个值不想等

  • b1 lt b2  判断b1小于(less than)b2

  • b1 gt b2  判断b1小于(greater than)b2

  • b1 lte b2:判断b1小于等于b2

  • b1 gte b2:判断b1大于等于b2

  • b1 in b2 判断b2包含b1

  • b1 not in b2  判断b2不包含b1

这些表达式经常和test配合。

四则运算赋值

还有一些表达式用来赋值或者增强属性。经常用来做模糊搜索的 bind标签:

<bind name="nameLike" value="'%'+ name + '%'"/>

  

这里的value也属于OGNL表达式 e1+e2,字符串是拼接,数字的话就是加法运算,我们可以引申出肯定还有:

  • e1*e2 乘法

  • e1/e2 除法

  • e1-e2 减法

  • e1%e2 取模

类的内置方法

其实MybatisMapper.xml中还可以使用对象的内置方法,比如我们需要判断一个java.util.Collection集合是否为空,可以这么写:


   
  1. <if test="collection!=null and collection.size()> 0">
  2.   and some_col = #{some_val}
  3. </if>

这里就使用了对象的内置方法Collection.size()

我们还可以调用自定义对象CollectionUtils静态方法来判断集合是否为空:


   
  1. package cn.felord.util;
  2. public final class CollectionUtils {
  3.  public static boolean isNotEmpty( Collection<?> collection) {
  4.   return (collection != null && !collection.isEmpty());
  5.  }    
  6. }

那么上面的<if>判断改为:


   
  1. <if test="@cn.felord.util.CollectionUtils@isNotEmpty(collection)">
  2.   and some_col = #{some_val}
  3. </if>

不要忘了这里要带上类的全限定名。

取值操作

取值操作的话,如果是对象直接e.property,如果是集合或者Map可以e[index|key],通过索引或者键名来取值。分别举个例子:


   
  1. # 对象取属性
  2. user.username
  3. # 集合取元素
  4. array[1
  5. map 取值
  6. map['username']

其实静态属性也能取值调用,跟上面的静态方法类似:

@cn.felord.Cache@user

  

对应Java代码:


   
  1. package cn.felord;
  2. public final class Cache {
  3.  public static User user = new User ("felord.cn") ;
  4. }

赋值操作

上面的取值除了可以做判断还可以用来SQL参数赋值:


   
  1.      <where>
  2.          <!-- 常用的赋值方式 -->
  3.              username = #{username}
  4.          <!-- $ 也可以赋值 -->
  5.              and user_id =${userId}
  6.          <!-- 对象取属性 -->
  7.              and id = ${user.id}
  8.          <!-- Math.abs  双@简写 -->
  9.              and age = ${@@abs(-12345678)}
  10.          <!-- 调用枚举 -->
  11.              and gender =${@cn.felord.GenderEnum@MALE.ordinal()}
  12.              and id=${@cn.felord.Cache@user.userId}
  13.          </where>

通过${}符号可以用OGNL表达式给SQL参数赋值,不过感觉平常比较少用。还有一些OGNL的 玩法可以去看官方文档。

总结

今天对Mybatis中的OGNL表达式进行了总结和分组,对常用的和不常用的用法进行了归纳,希望能够帮助你掌握Mybatis动态SQL的深度运用。不过请尽量将复杂的操作简单化,不要写过于复杂的OGNL表达式,无论是从性能上还是并发安全上都是很重要的因素。好了今天的分享就到这里,请多多关注:码农小胖哥,获取日常开发中有用的干货知识。如果你对OGNL的用法有自己的心得体会,欢迎留言讨论。

厉害了!Spring Boot 2.5正式发布

2021-05-21

Spring Security5.5发布,正式实装OAuth2.0的第五种授权模式

2021-05-19

文章来源: felord.blog.csdn.net,作者:码农小胖哥,版权归原作者所有,如需转载,请联系作者。

原文链接:felord.blog.csdn.net/article/details/117267913

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。