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

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

前言

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

Mybatis中的OGNL

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

<if test="field!='' and field!= null">  and some_col = #{field}</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集合是否为空,可以这么写:

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

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

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

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

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

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

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

取值操作

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

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

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

@cn.felord.Cache@user

对应Java代码:

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

赋值操作

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

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

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

转载于公众号-程序猿DD


北京云中融信网络科技有限公司(简称融云),是安全、可靠的全球互联网通信云服务商,向开发者和企业提供即时通讯和实时音视频通信云服务。iResearch 艾瑞权威数据报告显示,融云即时通讯云市场份额连续多年稳居头位。

融云构建了一张覆盖全球所有国家及地区(共 233 个)的通信云网络,在全球各地设立多个数据中心及数千个加速节点。基于客户业务需求,融云可提供多种部署模式——公有云、私有云、混合云,为全球企业提供稳定的互联网通信云服务。针对企业级用户,融云将业务垂直到各个行业,为社交、直播、金融、交通运输、教育、电商、医疗等多个行业领域推出了针对性解决方案。

融云基于海量业务的技术锤炼,从基础架构到精细化运营,充分体现平台实力;凭借卓越的产品和优质的服务,在开发者规模、行业覆盖率、平台日活跃用户数、日均消息量等方面超越全行业。目前,已有数十万互联网用户及上千家企业级用户通过融云实现了场景化沟通,并从中获益,包括工商银行、中国移动、四川航空、CCTV 微视、中联重科、58 赶集、大河报业、新东方、陆金所、融创地产、IDG、华兴资本、易车网、猪八戒、得到 APP、荔枝、汽车之家、哈啰出行、百姓网、StarMaker、Opera、Elelive。

       

标签: