Mybatis攔截器原理及實(shí)例

攔截器類型

1.Executor (update, query, flushStatements, commit, rollback, getTransaction, close, isClosed) 攔截執(zhí)行器的方法
2.ParameterHandler (getParameterObject, setParameters) 攔截參數(shù)的處理
2.ResultSetHandler (handleResultSets, handleOutputParameters) 攔截結(jié)果集的處理
4.StatementHandler (prepare, parameterize, batch, update, query) 攔截Sql語(yǔ)法和會(huì)話構(gòu)建的處理
執(zhí)行順序:Executor => StatementHandler => ParameterHandler => ResultSetHandler

攔截器應(yīng)用場(chǎng)景

StatementHandler可改寫sql 實(shí)現(xiàn)分頁(yè)查詢
ParameterHandler可改寫攔截參數(shù)的處理,新增創(chuàng)建時(shí)間、創(chuàng)建人、更新時(shí)間等
ResultSetHandler數(shù)據(jù)庫(kù)結(jié)果二次加密返回客戶端,過(guò)濾掉審計(jì)字段、敏感字段

攔截器增加分頁(yè)功能樣例

@Intercepts({@Signature(type = StatementHandler.class, method = "prepare",
                        args = {Connection.class, Integer.class}),
            })
@Component
@Slf4j
public class SqlInterceptor implements Interceptor {
    @Value("${jarye.pagehelper.rule}")
    private String pagehelperRule;
    
    @Override
    public Object intercept(Invocation invocation) throws Throwable {
        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語(yǔ)句是為查詢類型
            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) {
        
    }
    
    /**
    * 對(duì)sql實(shí)現(xiàn) 修改  加上limit
    */
    private void extendLimit(StatementHandler statementHandler) throws NoSuchFieldException, IllegalAccessException {
        // 獲取到原生sql語(yǔ)句
        BoundSql boundSql = statementHandler.getBoundSql();
        Class<? extends BoundSql> aClass = boundSql.getClass();
        // 使用反射機(jī)制修改原生sqk語(yǔ)句
        Field sql = aClass.getDeclaredField("sql");
        sql.setAccessible(true);
        String oldSqlStr = boundSql.getSql();
        // 后面加上 limit
        sql.set(boundSql, oldSqlStr + "   " + pagehelperRule);
    }
}

攔截器修改參數(shù)樣例

@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 {
        // 獲取攔截器攔截的設(shè)置參數(shù)對(duì)象DefaultParameterHandler
        ParameterHandler parameterHandler = (ParameterHandler) invocation.getTarget();

        // 通過(guò)mybatis的反射來(lái)獲取對(duì)應(yīng)的值
        MetaObject metaResultSetHandler = MetaObject.forObject(parameterHandler, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY, REFLECTOR_FACTORY);
        MappedStatement mappedStatement = (MappedStatement) metaResultSetHandler.getValue("mappedStatement");
        // 如果方法是無(wú)慘的,則parameterObject會(huì)存在空的情況
        Object parameterObject = metaResultSetHandler.getValue("parameterObject");
        //sql類型
        SqlCommandType sqlCommandType = mappedStatement.getSqlCommandType();
        // 回寫parameterObject對(duì)象
        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 {
        if(null==parameterObject){
            return parameterObject;
        }
        Class<?> aClass = parameterObject.getClass();
        switch (sqlCommandType) {
            case INSERT:
                // 使用反射獲取到createDatetime修改時(shí)間為當(dāng)前系統(tǒng)時(shí)間
                Field createDatetime = aClass.getSuperclass().getDeclaredField("createDatetime");
                createDatetime.setAccessible(true);
                createDatetime.set(parameterObject, new Date());
            case UPDATE:
                //updateDatetime字段 修改時(shí)間為當(dāng)前系統(tǒng)時(shí)間
                Field updateDatetime = aClass.getSuperclass().getDeclaredField("updateDatetime");
                updateDatetime.setAccessible(true);
                updateDatetime.set(parameterObject, new Date());
                break;
        }
        return parameterObject;
    }
}

推薦閱讀:
<<<Mybatis的整體執(zhí)行原理圖解
<<<SqlSessionFactory的創(chuàng)建過(guò)程原理
<<<SqlSession的創(chuàng)建過(guò)程
<<<sqlSession如何獲得具體的Mapper接口信息
<<<userMapper.getUser(1);底層實(shí)現(xiàn)原理
<<<sqlSession.selectOne底層實(shí)現(xiàn)原理
<<<Mybatis一級(jí)緩存知識(shí)匯總
<<<Mybatis二級(jí)緩存知識(shí)匯總
<<<Springboot整合Mybatis二級(jí)緩存
<<<Mybatis常見(jiàn)面試題

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時(shí)請(qǐng)結(jié)合常識(shí)與多方信息審慎甄別。
平臺(tái)聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡(jiǎn)書系信息發(fā)布平臺(tái),僅提供信息存儲(chǔ)服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容