mybatis拦截器怎么使用

其他教程   发布日期:2023年09月23日   浏览次数:412

今天小编给大家分享一下mybatis拦截器怎么使用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。

    mybatis实战之拦截器

    在服务的开发过程中,往往存在这样的需求,针对业务,实现对数据库操作语句做统一的处理。

    比如对某些敏感数据如用户姓名、手机号等坐脱敏处理保存和查询、对未实现权限的查询通过添加关联查询实现权限控制查询结果等等。

    这时,mybatis框架提供了拦截器的方式,允许在映射语句执行过程中的某一点进行拦截调用,进行自己的业务处理。

    1、使用方法

    这里参考了官网的使用说明,只需实现 Interceptor 接口,并在类中指定想要拦截的方法签名即可。

    比如:

    1. @Intercepts({@Signature(
    2. type= Executor.class,
    3. method = "update",
    4. args = {MappedStatement.class,Object.class})})
    5. public class ExamplePlugin implements Interceptor {
    6. private Properties properties = new Properties();
    7. public Object intercept(Invocation invocation) throws Throwable {
    8. // implement pre processing if need
    9. Object returnObject = invocation.proceed();
    10. // implement post processing if need
    11. return returnObject;
    12. }
    13. public void setProperties(Properties properties) {
    14. this.properties = properties;
    15. }
    16. }

    然后在mybatis的配置文件中,添加插件的对应配置即可。

    1. <!-- mybatis-config.xml -->
    2. <plugins>
    3. <plugin interceptor="org.mybatis.example.ExamplePlugin">
    4. <property name="someProperty" value="100"/>
    5. </plugin>
    6. </plugins>

    我们也可以在代码中添加,下面给出在spring中

    1. //通过spring查找SqlSessionFactory对象的逻辑在此省略
    2. SqlSessionFactory sqlSessionFactory = (SqlSessionFactory) var3;
    3. org.apache.ibatis.session.Configuration c = sqlSessionFactory.getConfiguration();
    4. c.addInterceptor(interceptor);

    分别加上处理的业务逻辑,这个拦截器就可以使用了。

    2、需要注意的地方

    第一节简单介绍了,拦截器的使用方法,但在实际项目中这样还远远不够。

    笔者在本节列举了一些需要注意的地方,供大家思考讨论。

    拦截器的执行顺序

    拦截器的调用顺序分为两大种,第一种是拦截的不同对象,第二种是指拦截同一种对象的同一个方法。

    第一种情况,例如拦截 Executor 和 拦截 StatementHandler 就属于不同的拦截对象, 这两类的拦截器在整体执行的逻辑上是不同的。

    StatementHandler 属于 Executor 执行过程中的一个子过程。

    所以这两种不同类别的插件在配置时,一定是先执行 Executor 的拦截器,然后才会轮到 StatementHandler。

    所以这种情况下配置拦截器的顺序就不重要了,在 MyBatis 逻辑上就已经控制了先后顺序。

    第二种情况,例如都拦截 Executor 的 query 方法,这时你配置拦截器的顺序就会对这里有影响了。比如配置如下。

    1. <plugins>
    2. <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor1"/>
    3. <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor2"/>
    4. <plugin interceptor="com.github.pagehelper.ExecutorQueryInterceptor3"/>
    5. </plugins>

    前面我们配置拦截器的顺序是1,2,3。在这里也会按照 1,2,3 的顺序被层层代理,代理后的结构如下:

    1. Interceptor3:{
    2. Interceptor2: {
    3. Interceptor1: {
    4. target: Executor
    5. }
    6. }
    7. }

    然后到执行的逻辑:

    Interceptor3 前置处理
    Interceptor2 前置处理
    Interceptor1 前置处理
    Object result = executor.query(4个参数方法);
    Interceptor1 后续处理
    Interceptor2 后续处理
    Interceptor3 后续处理
    return result;

    顺序就是 3>2>1>Executor>1>2>3。MyBatis的拦截器采用责任链设计模式,多个拦截器之间的责任链是通过动态代理组织的。

    我们一般都会在拦截器中的intercept方法中往往会有invocation.proceed()语句,其作用是将拦截器责任链向后传递,本质上便是动态代理的invoke。

    与常用插件的整合遇到的问题

    pageHelper造成分页失效的问题

    通过查看pagehelper源码,可以看到其inercept方法直接获取了excutor然后开始分页查询,当查询到结果时,便返回了。

    就是pagehelper的intercept方法中没有invocation.proceed(),这意味着什么?

    1. //com.github.pagehelper.PageInterceptor#intercept
    2. .....
    3. resultList = executor.query(ms, parameter, rowBounds, resultHandler, cacheKey, boundSql);
    4. }
    5. return dialect.afterPage(resultList, parameter, rowBounds);
    6. } finally {
    7. dialect.afterAll();
    8. }

    这意味着pagehelper没有继续向后传递责任链,而是自行处理直接返回。

    由此,我们可以猜出该问题大概率与拦截器的执行顺序有关。

    通过断点调试,验证了该猜想,当遇到分页查询时,执行到pagehelper就结束了,没有进入我们的自定义拦截器。这就可能造成我们自定义拦截器失效。

    解决方案

    因为PageHelper是Excetor类型的拦截器,所以我们如果想要在PageHelper拦截器前面执行,就必须要将我们自己的拦截器添加到他的拦截器后面。

    这里只介绍最简单最优雅的一种方式:

    注册一个ApplicationListener监听器,监听 ContextRefreshedEvent 事件,当所有的bean都初始化完成后(即PageHelper也已经注册好了),再把我们的自定义 MyBatis 拦截器注册到 SqlSessionFactory 中。

    可以提升的点

    Interceptor接口提供了三个方法分别是拦截器处理逻辑的主要方法、判断是否要进行拦截,然后做出决定是否生成一个代理的方法及设置参数的方法。

    1. package org.apache.ibatis.plugin;
    2. import java.util.Properties;
    3. public interface Interceptor {
    4. Object intercept(Invocation invocation) throws Throwable;
    5. default Object plugin(Object target) {
    6. return Plugin.wrap(target, this);
    7. }
    8. default void setProperties(Properties properties) {
    9. // NOP
    10. }
    11. }

    这里说的提升点,就是在实现接口的实现类中,我们可以在plugin方法里加上一个判断,因为默认情况下,拦截器根据顺序拦截后,就可以去处理对应逻辑了,这里加上一个判断拦截的条件,可以减少代理类的创建。

    1. @Override
    2. public Object plugin(Object target) {
    3. if (target instanceof StatementHandler && checkIfNeeded((StatementHandler) target)) {
    4. return Plugin.wrap(target, this);
    5. } else {
    6. return target;
    7. }
    8. }

    以上就是mybatis拦截器怎么使用的详细内容,更多关于mybatis拦截器怎么使用的资料请关注九品源码其它相关文章!