改造Cache插件

在前面的文章里澈灼,介紹了兩個(gè)插件:根據(jù)注解實(shí)現(xiàn)的sql自動(dòng)生成插件和分頁(yè)插件。這兩個(gè)插件在沒(méi)有開(kāi)啟cache的情況下可以很好的使用,但開(kāi)啟cache后卻出現(xiàn)了一些問(wèn)題,為了解決這些問(wèn)題汽绢,編寫(xiě)了攔截cache的插件,通過(guò)這個(gè)攔截器修正了這些問(wèn)題侧戴。

1 問(wèn)題#

1.1 什么問(wèn)題##

最容易出現(xiàn)的問(wèn)題是開(kāi)啟cache后庶喜,分頁(yè)查詢(xún)時(shí)無(wú)論查詢(xún)哪一頁(yè)都返回第一頁(yè)的數(shù)據(jù)。另外救鲤,使用sql自動(dòng)生成插件生成get方法的sql時(shí),傳入的參數(shù)不起作用秩冈,無(wú)論傳入的參數(shù)是多少本缠,都返回第一個(gè)參數(shù)的查詢(xún)結(jié)果。

1.2 為什么出現(xiàn)這些問(wèn)題##

在之前講解Mybatis的執(zhí)行流程的時(shí)候提到入问,在開(kāi)啟cache的前提下丹锹,Mybatis的executor會(huì)先從緩存里讀取數(shù)據(jù)稀颁,讀取不到才去數(shù)據(jù)庫(kù)查詢(xún)。問(wèn)題就出在這里楣黍,sql自動(dòng)生成插件和分頁(yè)插件執(zhí)行的時(shí)機(jī)是在statementhandler里匾灶,而statementhandler是在executor之后執(zhí)行的,無(wú)論sql自動(dòng)生成插件和分頁(yè)插件都是通過(guò)改寫(xiě)sql來(lái)實(shí)現(xiàn)的租漂,executor在生成讀取cache的key(key由sql以及對(duì)應(yīng)的參數(shù)值構(gòu)成)時(shí)使用都是原始的sql阶女,這樣當(dāng)然就出問(wèn)題了。

1.3 解決問(wèn)題##

找到問(wèn)題的原因后哩治,解決起來(lái)就方便了秃踩。只要通過(guò)攔截器改寫(xiě)executor里生成key的方法,在生成可以時(shí)使用自動(dòng)生成的sql(對(duì)應(yīng)sql自動(dòng)生成插件)或加入分頁(yè)信息(對(duì)應(yīng)分頁(yè)插件)就可以了业筏。

2 攔截器簽名#

@Intercepts({@Signature(type = Executor.class, method = "query", args = {MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class})})  
public class CacheInterceptor implements Interceptor {  
    ...  
}

從簽名里可以看出憔杨,要攔截的目標(biāo)類(lèi)型是Executor(注意:type只能配置成接口類(lèi)型),攔截的方法是名稱(chēng)為query的方法蒜胖。

3 intercept實(shí)現(xiàn)#

public Object intercept(Invocation invocation) throws Throwable {  
        Executor executorProxy = (Executor) invocation.getTarget();  
        MetaObject metaExecutor = MetaObject.forObject(executorProxy, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
        // 分離代理對(duì)象鏈  
        while (metaExecutor.hasGetter("h")) {  
            Object object = metaExecutor.getValue("h");  
            metaExecutor = MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
        }  
        // 分離最后一個(gè)代理對(duì)象的目標(biāo)類(lèi)  
        while (metaExecutor.hasGetter("target")) {  
            Object object = metaExecutor.getValue("target");  
            metaExecutor = MetaObject.forObject(object, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
        }  
        Object[] args = invocation.getArgs();  
        return this.query(metaExecutor, args);  
    }  
  
    public <E> List<E> query(MetaObject metaExecutor, Object[] args) throws SQLException {  
        MappedStatement ms = (MappedStatement) args[0];  
        Object parameterObject = args[1];  
        RowBounds rowBounds = (RowBounds) args[2];  
        ResultHandler resultHandler = (ResultHandler) args[3];  
        BoundSql boundSql = ms.getBoundSql(parameterObject);  
        // 改寫(xiě)key的生成  
        CacheKey cacheKey = createCacheKey(ms, parameterObject, rowBounds, boundSql);  
        Executor executor = (Executor) metaExecutor.getOriginalObject();  
        return executor.query(ms, parameterObject, rowBounds, resultHandler, cacheKey, boundSql);  
    }  
  
    private CacheKey createCacheKey(MappedStatement ms, Object parameterObject, RowBounds rowBounds, BoundSql boundSql) {  
        Configuration configuration = ms.getConfiguration();  
        pageSqlId = configuration.getVariables().getProperty("pageSqlId");  
        if (null == pageSqlId || "".equals(pageSqlId)) {  
            logger.warn("Property pageSqlId is not setted,use default '.*Page$' ");  
            pageSqlId = defaultPageSqlId;  
        }  
        CacheKey cacheKey = new CacheKey();  
        cacheKey.update(ms.getId());  
        cacheKey.update(rowBounds.getOffset());  
        cacheKey.update(rowBounds.getLimit());  
        List<ParameterMapping> parameterMappings = boundSql.getParameterMappings();  
        // 解決自動(dòng)生成SQL消别,SQL語(yǔ)句為空導(dǎo)致key生成錯(cuò)誤的bug  
        if (null == boundSql.getSql() || "".equals(boundSql.getSql())) {  
            String id = ms.getId();  
            id = id.substring(id.lastIndexOf(".") + 1);  
            String newSql = null;  
            try {  
                if ("select".equals(id)) {  
                    newSql = SqlBuilder.buildSelectSql(parameterObject);  
                }  
                SqlSource sqlSource = buildSqlSource(configuration, newSql, parameterObject.getClass());  
                parameterMappings = sqlSource.getBoundSql(parameterObject).getParameterMappings();  
                cacheKey.update(newSql);  
            } catch (Exception e) {  
                logger.error("Update cacheKey error.", e);  
            }  
        } else {  
            cacheKey.update(boundSql.getSql());  
        }  
  
        MetaObject metaObject = MetaObject.forObject(parameterObject, DEFAULT_OBJECT_FACTORY, DEFAULT_OBJECT_WRAPPER_FACTORY);  
  
        if (parameterMappings.size() > 0 && parameterObject != null) {  
            TypeHandlerRegistry typeHandlerRegistry = ms.getConfiguration().getTypeHandlerRegistry();  
            if (typeHandlerRegistry.hasTypeHandler(parameterObject.getClass())) {  
                cacheKey.update(parameterObject);  
            } else {  
                for (ParameterMapping parameterMapping : parameterMappings) {  
                    String propertyName = parameterMapping.getProperty();  
                    if (metaObject.hasGetter(propertyName)) {  
                        cacheKey.update(metaObject.getValue(propertyName));  
                    } else if (boundSql.hasAdditionalParameter(propertyName)) {  
                        cacheKey.update(boundSql.getAdditionalParameter(propertyName));  
                    }  
                }  
            }  
        }  
        // 當(dāng)需要分頁(yè)查詢(xún)時(shí),將page參數(shù)里的當(dāng)前頁(yè)和每頁(yè)數(shù)加到cachekey里  
        if (ms.getId().matches(pageSqlId) && metaObject.hasGetter("page")) {  
            PageParameter page = (PageParameter) metaObject.getValue("page");  
            if (null != page) {  
                cacheKey.update(page.getCurrentPage());  
                cacheKey.update(page.getPageSize());  
            }  
        }  
        return cacheKey;  
}

4 plugin實(shí)現(xiàn)#

public Object plugin(Object target) {  
    // 當(dāng)目標(biāo)類(lèi)是CachingExecutor類(lèi)型時(shí)台谢,才包裝目標(biāo)類(lèi)寻狂,否者直接返回目標(biāo)本身,減少目標(biāo)被代理的  
    // 次數(shù)  
    if (target instanceof CachingExecutor) {  
        return Plugin.wrap(target, this);  
    } else {  
        return target;  
    }  
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市对碌,隨后出現(xiàn)的幾起案子荆虱,更是在濱河造成了極大的恐慌,老刑警劉巖朽们,帶你破解...
    沈念sama閱讀 211,194評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件怀读,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡骑脱,警方通過(guò)查閱死者的電腦和手機(jī)菜枷,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評(píng)論 2 385
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)叁丧,“玉大人啤誊,你說(shuō)我怎么就攤上這事∮德Γ” “怎么了蚊锹?”我有些...
    開(kāi)封第一講書(shū)人閱讀 156,780評(píng)論 0 346
  • 文/不壞的土叔 我叫張陵,是天一觀(guān)的道長(zhǎng)稚瘾。 經(jīng)常有香客問(wèn)我牡昆,道長(zhǎng),這世上最難降的妖魔是什么摊欠? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,388評(píng)論 1 283
  • 正文 為了忘掉前任丢烘,我火速辦了婚禮柱宦,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘播瞳。我一直安慰自己掸刊,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,430評(píng)論 5 384
  • 文/花漫 我一把揭開(kāi)白布赢乓。 她就那樣靜靜地躺著忧侧,像睡著了一般。 火紅的嫁衣襯著肌膚如雪骏全。 梳的紋絲不亂的頭發(fā)上苍柏,一...
    開(kāi)封第一講書(shū)人閱讀 49,764評(píng)論 1 290
  • 那天,我揣著相機(jī)與錄音姜贡,去河邊找鬼试吁。 笑死,一個(gè)胖子當(dāng)著我的面吹牛楼咳,可吹牛的內(nèi)容都是我干的熄捍。 我是一名探鬼主播,決...
    沈念sama閱讀 38,907評(píng)論 3 406
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼母怜,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼余耽!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起苹熏,我...
    開(kāi)封第一講書(shū)人閱讀 37,679評(píng)論 0 266
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤碟贾,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后轨域,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體袱耽,經(jīng)...
    沈念sama閱讀 44,122評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,459評(píng)論 2 325
  • 正文 我和宋清朗相戀三年干发,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了朱巨。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,605評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡枉长,死狀恐怖冀续,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情必峰,我是刑警寧澤洪唐,帶...
    沈念sama閱讀 34,270評(píng)論 4 329
  • 正文 年R本政府宣布,位于F島的核電站吼蚁,受9級(jí)特大地震影響凭需,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,867評(píng)論 3 312
  • 文/蒙蒙 一功炮、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧术唬,春花似錦薪伏、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,734評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至借浊,卻和暖如春塘淑,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背蚂斤。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,961評(píng)論 1 265
  • 我被黑心中介騙來(lái)泰國(guó)打工存捺, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人曙蒸。 一個(gè)月前我還...
    沈念sama閱讀 46,297評(píng)論 2 360
  • 正文 我出身青樓捌治,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親纽窟。 傳聞我的和親對(duì)象是個(gè)殘疾皇子肖油,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,472評(píng)論 2 348

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