• 14.mybatis拦截器原理


    mybatis拦截器介绍

    MyBatis 允许你在映射语句执行过程中的某一点进行拦截调用,默认情况下,MyBatis 允许使用插件来拦截的方法调用包括:

    1.Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed) 拦截执行器的方法

    2.ParameterHandler (getParameterObject, setParameters) 拦截参数的处理

    2.ResultSetHandler (handleResultSets, handleOutputParameters) 拦截结果集的处理

    4.StatementHandler (prepare, parameterize, batch, update, query) 拦截Sql语法和会话构建的处理

    执行过程中按照Executor => StatementHandler => ParameterHandler => ResultSetHandler。

    Executor在执行过程中会创建StatementHandler,在创建StatementHandler过程中会创建 ParameterHandler和ResultSetHandler。

    StatementHandler改写sql 实现分页插件

    ParameterHandler 改写拦截参数的处理

    ResultSetHandler 数据结果 数据库结果二次加密返回客户端

    Executor

    StatementHandler

    ParameterHandler

    ResultSetHandler

    mybatis拦截器应用场景

    拦截器一般在业务处理中用于:

    1.分页查询

    2.对返回结果,过滤掉审计字段,敏感字段

    3.对返回结果中的加密数据进行解密

    4.对新增数据自动添加创建人,创建时间,更新时间,更新人 ,对更新数据自动新增更新时间,更新人

    mybatis拦截器实现分页

    package com.mayikt.config;
    
    import lombok.extern.slf4j.Slf4j;
    import org.apache.ibatis.executor.statement.StatementHandler;
    import org.apache.ibatis.mapping.BoundSql;
    import org.apache.ibatis.mapping.MappedStatement;
    import org.apache.ibatis.mapping.SqlCommandType;
    import org.apache.ibatis.plugin.*;
    import org.apache.ibatis.reflection.MetaObject;
    import org.apache.ibatis.reflection.SystemMetaObject;
    import org.springframework.beans.factory.annotation.Value;
    import org.springframework.stereotype.Component;
    
    import java.lang.reflect.Field;
    import java.sql.Connection;
    import java.util.Properties;
    
    /**
    * 拦截Sql语法和会话构建的处理
    */
    @Intercepts({@Signature(type = StatementHandler.class, method = "prepare",
                            args = {Connection.class, Integer.class}),
                })
    @Component
    @Slf4j
    public class MybatisSqlInterceptor implements Interceptor {
        @Value("${mayikt.pagehelper.rule}")
        private String pagehelperRule;
        
        @Override
        public Object intercept(Invocation invocation) throws Throwable {
            log.info("");
            StatementHandler statementHandler = (StatementHandler) invocation.getTarget();
            MetaObject metaObject = SystemMetaObject.forObject(statementHandler);
            MappedStatement mappedStatement = (MappedStatement) metaObject.getValue("delegate.mappedStatement");
            //sql类型
            SqlCommandType sqlCommandType = mappedStatement.getSqlCommandType();
            switch (sqlCommandType) {
                    // 判断sql语句是为查询类型
                case SELECT:
                    extendLimit(statementHandler);
                    break;
            }
            
            
            return invocation.proceed();
        }
        
        @Override
        public Object plugin(Object target) {
            return Plugin.wrap(target, this);
        }
        
        
        @Override
        public void setProperties(Properties properties) {
            
        }
        
        /**
        * 对sql实现 修改  加上limit
        */
        private void extendLimit(StatementHandler statementHandler) throws NoSuchFieldException, IllegalAccessException {
            // 获取到原生sql语句
            BoundSql boundSql = statementHandler.getBoundSql();
            Class<? extends BoundSql> aClass = boundSql.getClass();
            // 使用反射机制修改原生sqk语句
            Field sql = aClass.getDeclaredField("sql");
            sql.setAccessible(true);
            String oldSqlStr = boundSql.getSql();
            // 后面加上 limit
            sql.set(boundSql, oldSqlStr + "   " + pagehelperRule);
        }
        
        
    }
    mayikt:
      pagehelper:
       rule: limit 2
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79

    mybatis拦截器自动添加创建时间,更新时间

    package com.mayikt.config;
    
    import lombok.extern.slf4j.Slf4j;
    import org.apache.ibatis.executor.parameter.ParameterHandler;
    import org.apache.ibatis.mapping.BoundSql;
    import org.apache.ibatis.mapping.MappedStatement;
    import org.apache.ibatis.mapping.SqlCommandType;
    import org.apache.ibatis.plugin.Interceptor;
    import org.apache.ibatis.plugin.Intercepts;
    import org.apache.ibatis.plugin.Invocation;
    import org.apache.ibatis.plugin.Plugin;
    import org.apache.ibatis.plugin.Signature;
    import org.apache.ibatis.reflection.DefaultReflectorFactory;
    import org.apache.ibatis.reflection.MetaObject;
    import org.apache.ibatis.reflection.ReflectorFactory;
    import org.apache.ibatis.reflection.factory.DefaultObjectFactory;
    import org.apache.ibatis.reflection.factory.ObjectFactory;
    import org.apache.ibatis.reflection.wrapper.DefaultObjectWrapperFactory;
    import org.apache.ibatis.reflection.wrapper.ObjectWrapperFactory;
    import org.springframework.stereotype.Component;
    
    import java.lang.reflect.Field;
    import java.lang.reflect.Method;
    import java.sql.PreparedStatement;
    import java.util.Date;
    import java.util.Properties;
    
    @Intercepts({
            @Signature(type = ParameterHandler.class, method = "setParameters", args = PreparedStatement.class)
    })
    @Slf4j
    @Component
    public class ParamInterceptor implements Interceptor {
    
    
        private static final ObjectFactory DEFAULT_OBJECT_FACTORY = new DefaultObjectFactory();
        private static final ObjectWrapperFactory DEFAULT_OBJECT_WRAPPER_FACTORY = new DefaultObjectWrapperFactory();
        private static final ReflectorFactory REFLECTOR_FACTORY = new DefaultReflectorFactory();
    
        @Override
        public Object intercept(Invocation invocation) throws Throwable {
            log.info("");
            // 获取拦截器拦截的设置参数对象DefaultParameterHandler
            ParameterHandler parameterHandler = (ParameterHandler) invocation.getTarget();
    
            // 通过mybatis的反射来获取对应的值
            MetaObject metaResultSetHandler = MetaObject.forObject(parameterHandler, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY, REFLECTOR_FACTORY);
            MappedStatement mappedStatement = (MappedStatement) metaResultSetHandler.getValue("mappedStatement");
            Object parameterObject = metaResultSetHandler.getValue("parameterObject");
            //sql类型
            SqlCommandType sqlCommandType = mappedStatement.getSqlCommandType();
            // 回写parameterObject对象
            metaResultSetHandler.setValue("parameterObject", updateInsertParam(sqlCommandType, parameterObject));
            return invocation.proceed();
        }
    
        @Override
        public Object plugin(Object target) {
            return Plugin.wrap(target, this);
        }
    
        @Override
        public void setProperties(Properties properties) {
    
        }
    
        private Object updateInsertParam(SqlCommandType sqlCommandType, Object parameterObject) throws NoSuchFieldException, IllegalAccessException {
            Class<?> aClass = parameterObject.getClass();
            switch (sqlCommandType) {
                case INSERT:
                    // 使用反射获取到createDatetime修改时间为当前系统时间
                    Field createDatetime = aClass.getSuperclass().getDeclaredField("createDatetime");
                    createDatetime.setAccessible(true);
                    createDatetime.set(parameterObject, new Date());
                case UPDATE:
                    //updateDatetime字段 修改时间为当前系统时间
                    Field updateDatetime = aClass.getSuperclass().getDeclaredField("updateDatetime");
                    updateDatetime.setAccessible(true);
                    updateDatetime.set(parameterObject, new Date());
                    break;
            }
            return parameterObject;
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51
    • 52
    • 53
    • 54
    • 55
    • 56
    • 57
    • 58
    • 59
    • 60
    • 61
    • 62
    • 63
    • 64
    • 65
    • 66
    • 67
    • 68
    • 69
    • 70
    • 71
    • 72
    • 73
    • 74
    • 75
    • 76
    • 77
    • 78
    • 79
    • 80
    • 81
    • 82
    • 83
    • 84
  • 相关阅读:
    POI实现Excel导入和导出(源码测试)
    Linux---su:鉴定故障
    有了这一特性,不再安装脚手架,可创建Vue或React项目
    新材料企业ERP有几种?能帮助企业解决哪些问题
    云计算基础
    D. Chip Move(DP,优化时间和空间)
    强大的开源API接口可视化管理平台-YApi
    真无线蓝牙耳机哪款好?千元真无线蓝牙耳机推荐
    数据结构与算法完整版 | 超详细图解,看这一篇就够了
    ICCV2023 Tracking paper汇总(一)(多目标跟随、单目标跟随等)
  • 原文地址:https://blog.csdn.net/u014001523/article/details/126415923