前言

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的 玩法可以去看官方文档。

总结

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

往期推荐

用好 Spring AOP,天降大锅从容应对!

SpringBoot 部署 Jar 文件,瘦身优化指南 !

知乎高赞:那些学计算机的女生后来都怎么样了?

微信继续加持上班摸鱼功能!网友:离被开除更进一步...

共同创造最好的OS,openEuler Developer Day 报名通道开启

推荐关注本文作者:码农小胖哥

分享高质量编程知识,探讨IT人生

技术干货,实战技巧,面试技巧,前沿资讯一个都不能少

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

  1. mybatis02映射动态sql关联查询spring整合mybatis

    2019独角兽企业重金招聘Python工程师标准>>> 输入映射和输出映射: 动态sql: 关联查询_一对一: 关联查询_一对多: 一对一,一对多操作的区别: 一对一,resultM ...

  2. c++ 传入动态参数_一文了解Mybatis中动态SQL的实现

    一.动态SQL简介 MyBatis的强大特性之一便是它的动态 SQL.如果你有使用 JDBC 或其他类似框架的经验,你就能体会到根据不同条件拼接 SQL 语句有多么痛苦.拼接的时候要确保不能忘了必要的 ...

  3. MyBatis动态SQL底层原理分析 与 JavaScript中的Date对象,以及UTC、GMT、时区的关系...

    http://fangjian0423.github.io/categories/mybatis/ http://xtutu.me/the-date-object-in-js/

  4. 【MyBatis】学习纪要六:动态SQL

    2019独角兽企业重金招聘Python工程师标准>>> 引言 动态SQL:Dynamic SQL. 本节我们来通过 MyBatis 的官方文档进行学习. Description(描述 ...

  5. MyBatis-学习笔记08【08.动态SQL】

    Java后端 学习路线 笔记汇总表[黑马程序员] MyBatis-学习笔记01[01.Mybatis课程介绍及环境搭建][day01] MyBatis-学习笔记02[02.Mybatis入门案例] M ...

  6. Mybatis动态sql和分页

    mybatis动态sql 1.1 if1.2 trim1.3 foreach1.4 其他choose/set/where 动态sql代码展示 <select id="list1&quo ...

  7. mybatis的动态sql和分页

    先简单讲一下mybatis. mybatis是一个orm框架.持久层框架.作用于dao层,负责数据库的访问操作. 几乎消除了jdbc的赋值代码.动态获取结果集等等. 再来是应用的需求配置: 1)核心配 ...

  8. Oracle基础 动态SQL语句

    一.静态SQL和动态SQL的概念. 1.静态SQL 静态SQL是我们常用的使用SQL语句的方式,就是编写PL/SQL时,SQL语句已经编写好了.因为静态SQL是在编写程序时就确定了,我们只能使用SQL ...

  9. MyBatis动态SQL(认真看看, 以后写SQL就爽多了)

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试文章 来源:cnblogs.com/homejim/p/9909657. ...

最新文章

  1. AAAI | 深度生成模型—NEVAE
  2. 转【查看oracle数据库的连接数以及用户 】
  3. Java集合篇:ConcurrentHashMap详解(JDK1.8)
  4. chrome/chromium浏览器的Enter passwod to unlock your login keyring
  5. SAP cross distribution chain status在Fiori应用中的draft handling
  6. P3980-[NOI2008]志愿者招募【费用流】
  7. 《UNIXLinux程序设计教程》一2.1 UNIX 输入输出基本概念
  8. 圆角边框(HTML、CSS)
  9. android jni示例_Android动画示例
  10. Node.js之Stream可读流readable
  11. 笔记(八)Jetson Nano 跑通 jetson-inference
  12. 基于ISIS路由协议的路由聚合
  13. echarts柱形图根据数据排序顺序要求更改颜色
  14. ufs 固态硬盘_看够了UFS/eMMC纠纷 再看笔记本固态硬盘速度的区别吧
  15. 使用Android SDK创建安卓虚拟机教程(Windows)
  16. android: Apostrophe not preceded by \ 错误
  17. 阿里云服务器购买过程中必须了解的注意事项
  18. 可穿戴市场掘金:VC如何看上小小的ZEPP公司
  19. 索尼开发新传感器为激光雷达提供助力,用于自动驾驶和其他应用
  20. PHP 获取客户端的真实IP

热门文章

  1. C++随笔——虚拟继承
  2. Android技术经理+资深工程师+研发工程师-杭州
  3. 文本比较算法Ⅶ——线性空间求最长公共子序列的Nakatsu算法
  4. java utf-8文件处理bom头
  5. centos7 tomcat8 配置 java web环境 熵池不够大 启动慢问题
  6. linux 内核编译错误 Makefile:416: *** mixed implicit and normal rules: deprecated syntax
  7. linux grub修复 手动引导进入系统
  8. python3 编写守护进程程序思路
  9. RGBA和ARGB的区别
  10. linux 线程 进程经典文章