Mybatis攔截器原理及實例

攔截器類型

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語法和會話構(gòu)建的處理
執(zhí)行順序:Executor => StatementHandler => ParameterHandler => ResultSetHandler

攔截器應用場景

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

攔截器增加分頁功能樣例

@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語句是為查詢類型
            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實現(xiàn) 修改  加上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);
    }
}

攔截器修改參數(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ù)對象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");
        // 如果方法是無慘的堡掏,則parameterObject會存在空的情況
        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 {
        if(null==parameterObject){
            return parameterObject;
        }
        Class<?> aClass = parameterObject.getClass();
        switch (sqlCommandType) {
            case INSERT:
                // 使用反射獲取到createDatetime修改時間為當前系統(tǒng)時間
                Field createDatetime = aClass.getSuperclass().getDeclaredField("createDatetime");
                createDatetime.setAccessible(true);
                createDatetime.set(parameterObject, new Date());
            case UPDATE:
                //updateDatetime字段 修改時間為當前系統(tǒng)時間
                Field updateDatetime = aClass.getSuperclass().getDeclaredField("updateDatetime");
                updateDatetime.setAccessible(true);
                updateDatetime.set(parameterObject, new Date());
                break;
        }
        return parameterObject;
    }
}

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

?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末报亩,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子盈匾,更是在濱河造成了極大的恐慌,老刑警劉巖毕骡,帶你破解...
    沈念sama閱讀 206,968評論 6 482
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件削饵,死亡現(xiàn)場離奇詭異,居然都是意外死亡未巫,警方通過查閱死者的電腦和手機窿撬,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,601評論 2 382
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來叙凡,“玉大人劈伴,你說我怎么就攤上這事∠烈蹋” “怎么了宰啦?”我有些...
    開封第一講書人閱讀 153,220評論 0 344
  • 文/不壞的土叔 我叫張陵,是天一觀的道長饼拍。 經(jīng)常有香客問我赡模,道長,這世上最難降的妖魔是什么师抄? 我笑而不...
    開封第一講書人閱讀 55,416評論 1 279
  • 正文 為了忘掉前任漓柑,我火速辦了婚禮,結(jié)果婚禮上叨吮,老公的妹妹穿的比我還像新娘辆布。我一直安慰自己,他們只是感情好茶鉴,可當我...
    茶點故事閱讀 64,425評論 5 374
  • 文/花漫 我一把揭開白布锋玲。 她就那樣靜靜地躺著,像睡著了一般涵叮。 火紅的嫁衣襯著肌膚如雪惭蹂。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,144評論 1 285
  • 那天割粮,我揣著相機與錄音盾碗,去河邊找鬼。 笑死舀瓢,一個胖子當著我的面吹牛廷雅,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 38,432評論 3 401
  • 文/蒼蘭香墨 我猛地睜開眼航缀,長吁一口氣:“原來是場噩夢啊……” “哼商架!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起谬盐,我...
    開封第一講書人閱讀 37,088評論 0 261
  • 序言:老撾萬榮一對情侶失蹤甸私,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后飞傀,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體皇型,經(jīng)...
    沈念sama閱讀 43,586評論 1 300
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,028評論 2 325
  • 正文 我和宋清朗相戀三年砸烦,在試婚紗的時候發(fā)現(xiàn)自己被綠了弃鸦。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,137評論 1 334
  • 序言:一個原本活蹦亂跳的男人離奇死亡幢痘,死狀恐怖唬格,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情颜说,我是刑警寧澤购岗,帶...
    沈念sama閱讀 33,783評論 4 324
  • 正文 年R本政府宣布,位于F島的核電站门粪,受9級特大地震影響喊积,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜玄妈,卻給世界環(huán)境...
    茶點故事閱讀 39,343評論 3 307
  • 文/蒙蒙 一乾吻、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧拟蜻,春花似錦绎签、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,333評論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至搔扁,卻和暖如春擒权,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背阁谆。 一陣腳步聲響...
    開封第一講書人閱讀 31,559評論 1 262
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留愉老,地道東北人场绿。 一個月前我還...
    沈念sama閱讀 45,595評論 2 355
  • 正文 我出身青樓,卻偏偏與公主長得像嫉入,于是被迫代替她去往敵國和親焰盗。 傳聞我的和親對象是個殘疾皇子璧尸,可洞房花燭夜當晚...
    茶點故事閱讀 42,901評論 2 345

推薦閱讀更多精彩內(nèi)容