MyBatis中的OGNL教程_isea533的博客-程序员秘密

技术标签: OGNL  mybatis  Mybatis示例  Mybatis  

MyBatis中的OGNL教程

有些人可能不知道MyBatis中使用了OGNL,有些人知道用到了OGNL却不知道在MyBatis中如何使用,本文就是讲如何在MyBatis中使用OGNL。

如果我们搜索OGNL相关的内容,通常的结果都是和Struts有关的,你肯定搜不到和MyBatis有关的,虽然和Struts中的用法类似但是换种方式理解起来就有难度。

MyBatis常用OGNL表达式

  • e1 or e2
  • e1 and e2
  • e1 == e2,e1 eq e2
  • e1 != e2,e1 neq e2
  • e1 lt e2:小于
  • e1 lte e2:小于等于,其他gt(大于),gte(大于等于)
  • e1 in e2
  • e1 not in e2
  • e1 + e2,e1 * e2,e1/e2,e1 - e2,e1%e2
  • !e,not e:非,求反
  • e.method(args)调用对象方法
  • e.property对象属性值
  • e1[ e2 ]按索引取值,List,数组和Map
  • @[email protected](args)调用类的静态方法
  • @[email protected]调用类的静态字段值

上述内容只是合适在MyBatis中使用的OGNL表达式,完整的表达式点击这里

MyBatis中什么地方可以使用OGNL?

如果你看过深入了解MyBatis参数,也许会有印象,因为这篇博客中提到了OGNL和一些特殊用法。

如果没看过,建议找时间看看,上面这篇博客不是很容易理解,但是理解后会很有用。

MyBatis中可以使用OGNL的地方有两处:

  • 动态SQL表达式中
  • ${param}参数中

上面这两处地方在MyBatis中处理的时候都是使用OGNL处理的。

下面通过举例来说明这两种情况的用法。

1.动态SQL表达式中

例一,MySql like 查询:

<select id="xxx" ...>
    select id,name,... from country
    <where>
        <if test="name != null and name != ''">
            name like concat('%', #{name}, '%')
        </if>
    </where>
</select>

上面代码中test的值会使用OGNL计算结果。

例二,通用 like 查询:

<select id="xxx" ...>
    select id,name,... from country
    <bind name="nameLike" value="'%' + name + '%'"/>
    <where>
        <if test="name != null and name != ''">
            name like #{nameLike}
        </if>
    </where>
</select>

这里<bind>value值会使用OGNL计算。

注:对<bind参数的调用可以通过#{}${} 方式获取,#{}可以防止注入。

在通用Mapper中支持一种UUID的主键,在通用Mapper中的实现就是使用了<bind>标签,这个标签调用了一个静态方法,大概方法如下:

<bind name="username_bind" 
      value='@[email protected]().toString().replace("-", "")' />

这种方式虽然能自动调用静态方法,但是没法回写对应的属性值,因此使用时需要注意。

2.${param}参数中

上面like的例子中使用下面这种方式最简单

<select id="xxx" ...>
    select id,name,... from country
    <where>
        <if test="name != null and name != ''">
            name like '${'%' + name + '%'}'
        </if>
    </where>
</select>

这里注意写的是${'%' + name + '%'},而不是%${name}%,这两种方式的结果一样,但是处理过程不一样。

在MyBatis中处理${}的时候,只是使用OGNL计算这个结果值,然后替换SQL中对应的${xxx},OGNL处理的只是${这里的表达式}

这里表达式可以是OGNL支持的所有表达式,可以写的很复杂,可以调用静态方法返回值,也可以调用静态的属性值。

例子:使用OGNL实现单表的分表功能

上面说的是OGNL简单的使用方法。这里举个OGNL实现数据库分表的例子。

分表这个功能是通用Mapper中的新功能,允许在运行的时候指定一个表名,通过指定的表名对表进行操作。这个功能实现就是使用了OGNL。

首先并不是所有的表都需要该功能,因此定义了一个接口,当参数(接口方法只有实体类一个参数)对象继承该接口的时候,就允许使用动态表名。

public interface IDynamicTableName {
    

    /**
     * 获取动态表名 - 只要有返回值,不是null和'',就会用返回值作为表名
     *
     * @return
     */
    String getDynamicTableName();
}

然后在XML中写表名的时候使用:

<if test="@[email protected](_parameter) 
            and dynamicTableName != null 
            and dynamicTableName != ''">
    ${dynamicTableName}
</if>
<if test="@[email protected](_parameter) 
            or dynamicTableName == null 
            or dynamicTableName == ''">
    defaultTableName
</if>

由于我需要判断_parameter是否继承了IDynamicTableName接口,简单的写法已经无法实现,所以使用了静态方法,这两个方法如下:

/**
 * 判断参数是否支持动态表名
 *
 * @param parameter
 * @return true支持,false不支持
 */
public static boolean isDynamicParameter(Object parameter) {
    if (parameter != null && parameter instanceof IDynamicTableName) {
        return true;
    }
    return false;
}

/**
 * 判断参数是否b支持动态表名
 *
 * @param parameter
 * @return true不支持,false支持
 */
public static boolean isNotDynamicParameter(Object parameter) {
    return !isDynamicParameter(parameter);
}

根据<if>判断的结果来选择使用那个表名。

另外注意XML判断中有一个dynamicTableName,这个参数是根据getDynamicTableName方法得到的,MyBatis使用属性对应的getter方法来获取值,不是根据field来获取值。

最后

如果你真想了解MyBatis中的OGNL用法,自己多写几个例子测试玩玩,动手测试是一种好的学习方式。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/isea533/article/details/50061705

智能推荐

洛谷1231 教辅的组成(网络流)_Michael_GLF的博客-程序员秘密

biubiu~【题目分析】板啊。。。。。调了那么久啊。。。。。这道题与Dining一题几乎一模一样啊。。。。。写的时候还是忘了拆点。。。。。。调了好久才想起。建图顺序为练习册(答案)---书---书---答案(练习册)根据给出关系建边即可。【代码~】#include&amp;lt;bits/stdc++.h&amp;gt;using namespace std;const int...

jUnit单元测试,反射,注解_云别的博客-程序员秘密

Junit单元测试:* 测试分类: 1. 黑盒测试:不需要写代码,给输入值,看程序是否能够输出期望的值。 不关心代码,只注重输入与输出 2. 白盒测试:需要写代码的。关注程序具体的执行流程。必须关心代码* Junit使用:白盒测试 * 步骤: 1. 定义一个测试类(测试用例) * 建议: * 测试类名:被测试的类名Test CalculatorTe...

Netbeans 快捷键_Nayson_的博客-程序员秘密

1、Ctrl-Tab:在打开的文件之间切换;2、Ctrl-N:在当前打开的项目里新建文件;3、Ctrl-F:当前文件查找匹配的字符(支持正则);4、Ctrl-H:当前文件查找、替换匹配的字符(支持正则,这里正则显得比较重要);5、Ctrl-Shift-↑↓:光标所在行向上或者向下复制;6、Ctrl-[:光标所在的位置首尾标识符之间移动({},ul );

基于JSP+Servlet+MySQL的新闻博客发布系统(附论文)_2033673808的博客-程序员秘密

项目运行截图 本科 毕 业 设 计(论文) 题目:新闻发布及管理系统的设计与实现 专题题目: 本科 毕 业 设 计(论文)任 务 书题目:新闻发布及管理系统的设计与实现专题题目(若无专题则不填):原始依据(包括设计(论文)的工作基础、研究条件、应用环境、工作目的等):工作基础基于web的新闻发布及管理系统的设计与实现,是动态网页和数据库结合,通过...

Fully Convolutional Network with Multi-Step Reinforcement Learning_multi_step_reinforcement_learning_我叫李劭卓的博客-程序员秘密

文章目录Fully Convolutional Network with Multi-Step Reinforcement Learningfor Image Processing(基于多步强化学习的全卷积网络图像处理 )0 Abstract1 Contribution2 Background3 Reinforcement Learning with Pixel-wise Rewards (PixelRL)(用像素激励强化学习)4 Reward Map Convolution (激励映射卷积)5 Appl

随便推点

深入了解一下Git-01: log和git reflog的区别_get reflog_goog_man的博客-程序员秘密

1、git loggit log 命令可以显示所有提交过的版本信息,不包括已经被删除的 commit 记录和 reset 的操作。如果感觉太繁琐,可以加上参数 --pretty=oneline,只会显示版本号和提交时的备注信息2、git refloggit reflog 可以查看所有分支的所有操作记录(包括已经被删除的 commit 记录和 reset 的操作),git reflog常用于恢复本地的错误操作。例如执行 git reset --hard HEAD~1,退回到上一个版本,用git

Hive 中时间,日期函数操作等总结_含着花的二哈的博客-程序员秘密

Hive 中时间,日期函数操作等的总结:1 获取当前时间 unix_timestamp()返回值: bigint说明: 获得当前时区的UNIX时间戳例子:hive (dwa)&gt; select unix_timestamp();OK16087990472 格式化时间 from_unixtime(timestamp,‘formatString’)返回值: string说明: 把UNIX时间戳转换成你想要的格式 , 默认是 yyyy-MM-dd HH:mm:ss例子1 :

IntelliJ IDEA安装.ignore插件忽略不必要提交的文件_LuckyJiang.2021的博客-程序员秘密

使用Git + IntelliJ IDEA提交项目到本地仓库的时候,会把.idea文件夹中的内容也提交上去,这目录里放置的是一些项目额配置信息,包括历史记录、版本控制信息等,是可以不传到Git上面去的。这个时候就需要编写.gitignore文件来忽略提交这些文件。在IDEA中有一个.ignore插件可以帮助我们做这件事情。接下来介绍如何安装并使用.ignore插件。一、安装点击Fi...

SQL Server占用内存过高及AWE的相关说明_awe占用内存过高_ghlfllz的博客-程序员秘密

SQL Server占用内存过高及AWE的相关说明开机SQL Server使用一段时间以后,内存会慢慢往上增长,最终会增长到系统内存的85%左右,听起来像是SQL Server有严重的内存泄露,其实不然,通过如下方式可以有效解决并控制SQL Server占用内存过高的问题。Step1. 登录SQL Server Management StudioStep2. 右键我们的

android键盘表情流畅切换实现_大泽哥的博客-程序员秘密

android 键盘表情流畅切换大家都用微信,在聊天页面,软键盘与表情栏,工具栏之间的切换非常流畅,没有引起输入bar的上下抖动,体验非常好。这里讲一讲它的实现原理。

Android 开发中使用Fragment跳转到下一个Fragment(使用add方法),并实现返回键和自定义返回键。_老咪MYC的博客-程序员秘密

Android 开发中使用Fragment跳转到下一个Fragment,并实现返回键和自定义返回键。项目结构:主类:MainActivityFirstFragmentSecondFragment辅助文件:BackHandledInterface.javaBackHandledFragment .java实现逻辑  入口:MainActivity中包含

推荐文章

热门文章

相关标签