日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術(shù)文章
文章詳情頁

淺談myBatis中的插件機(jī)制

瀏覽:197日期:2023-10-21 13:01:21

插件的配置與使用

在mybatis-config.xml配置文件中配置plugin結(jié)點(diǎn),比如配置一個(gè)自定義的日志插件LogInterceptor和一個(gè)開源的分頁插件PageInterceptor:

<plugins> <plugin interceptor='com.crx.plugindemo.LogInterceptor'></plugin> <plugin interceptor='com.github.pagehelper.PageInterceptor'> <property name='helperDialect' value='oracle' /> </plugin></plugins>

插件的工作原理

借助責(zé)任鏈模式,定義一系列的過濾器,在查詢等方法執(zhí)行時(shí)進(jìn)行過濾,從而達(dá)到控制參數(shù)、調(diào)整查詢語句和控制查詢結(jié)果等作用。下面從插件的加載(初始化)、注冊(cè)和調(diào)用這三個(gè)方面闡述插件的工作原理。

過濾器的加載(初始化)

和其他配置信息一樣,過濾器的加載也會(huì)在myBatis讀取配置文件創(chuàng)建Configuration對(duì)象時(shí)進(jìn)行,相應(yīng)的信息存儲(chǔ)在Configuration的interceptorChain屬性中,InterceptorChain封裝了一個(gè)包含Interceptor的list:

private final List<Interceptor> interceptors = new ArrayList<>();

在XMLConfigBuilder進(jìn)行解析配置文件時(shí)執(zhí)行pluginElement方法,生成過濾器實(shí)例,并添加到上述list中:

private void pluginElement(XNode parent) throws Exception { if (parent != null) { for (XNode child : parent.getChildren()) { String interceptor = child.getStringAttribute('interceptor'); Properties properties = child.getChildrenAsProperties(); Interceptor interceptorInstance = (Interceptor) resolveClass(interceptor).getDeclaredConstructor().newInstance(); interceptorInstance.setProperties(properties); configuration.addInterceptor(interceptorInstance); } }}

過濾器的注冊(cè)

可以為Executor、ParameterHandler、ResultSetHandler和StatementHandler四個(gè)接口注冊(cè)過濾器,注冊(cè)的時(shí)機(jī)也就是這四種接口的實(shí)現(xiàn)類的對(duì)象的生成時(shí)機(jī),比如Executor的過濾器的注冊(cè)發(fā)生在SqlSessionFactory使用openSession方法構(gòu)建SqlSession的過程中(因?yàn)镾qlSession依賴一個(gè)Executor實(shí)例),ParameterHandler和StatementHandler的過濾器發(fā)生在doQuery等sql執(zhí)行方法執(zhí)行時(shí)注冊(cè),而ResultHandler的過濾器的注冊(cè)則發(fā)生在查詢結(jié)果返回給客戶端的過程中。以Executor的過濾器的注冊(cè)為例,經(jīng)過了這樣的過程:

淺談myBatis中的插件機(jī)制

現(xiàn)在詳細(xì)的分析一下Plugin的wrap這個(gè)靜態(tài)的包裝方法:

public static Object wrap(Object target, Interceptor interceptor) { // 從定義的Interceptor實(shí)現(xiàn)類上的注解讀取需要攔截的類、方法 Map<Class<?>, Set<Method>> signatureMap = getSignatureMap(interceptor); // Executor、ParameterHandler、ResultSetHandler、StatementHandler Class<?> type = target.getClass(); // 從當(dāng)前執(zhí)行的目標(biāo)類中進(jìn)行匹配,過濾出符合當(dāng)前目標(biāo)的的過濾器 Class<?>[] interfaces = getAllInterfaces(type, signatureMap); if (interfaces.length > 0) { // 動(dòng)態(tài)代理生成Executor的代理實(shí)例 return Proxy.newProxyInstance(type.getClassLoader(), interfaces, new Plugin(target, interceptor, signatureMap)); } return target;}

上述代碼中的getSignatureMap方法是解析Interceptor上面的注解的過程,從注解中讀取出需要攔截的方法,依據(jù)@Signature的三個(gè)變量類、方法method和參數(shù)args就能通過反射唯一的定位一個(gè)需要攔截的方法。

private static Map<Class<?>, Set<Method>> getSignatureMap(Interceptor interceptor) { Intercepts interceptsAnnotation = interceptor.getClass().getAnnotation(Intercepts.class); if (interceptsAnnotation == null) { throw new PluginException( 'No @Intercepts annotation was found in interceptor ' + interceptor.getClass().getName()); } Signature[] sigs = interceptsAnnotation.value(); Map<Class<?>, Set<Method>> signatureMap = new HashMap<>(); for (Signature sig : sigs) { Set<Method> methods = signatureMap.computeIfAbsent(sig.type(), k -> new HashSet<>()); try { Method method = sig.type().getMethod(sig.method(), sig.args()); methods.add(method); } catch (NoSuchMethodException e) { throw new PluginException('Could not find method on ' + sig.type() + ' named ' + sig.method() + '. Cause: ' + e, e); } } return signatureMap;}

而getAllInterfaces方法是依據(jù)不同的目標(biāo)對(duì)象(Executor等四種)進(jìn)行過濾的過程,只給對(duì)應(yīng)的目標(biāo)進(jìn)行注冊(cè):

private static Class<?>[] getAllInterfaces(Class<?> type, Map<Class<?>, Set<Method>> signatureMap) { Set<Class<?>> interfaces = new HashSet<>(); while (type != null) { for (Class<?> c : type.getInterfaces()) { if (signatureMap.containsKey(c)) {interfaces.add(c); } } type = type.getSuperclass(); } return interfaces.toArray(new Class<?>[interfaces.size()]);}

至此,實(shí)際使用的Executor對(duì)象將是通過動(dòng)態(tài)代理生成的Plugin實(shí)例。

過濾器的調(diào)用

在第二步中完成了過濾器的注冊(cè),在實(shí)際調(diào)用Executor時(shí),將由實(shí)現(xiàn)了InvocationHandler接口的Plugin實(shí)例進(jìn)行接管,對(duì)Executor相應(yīng)方法方法的調(diào)用,將實(shí)際上調(diào)用動(dòng)態(tài)代理體系下的invoke方法:

public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { try { Set<Method> methods = signatureMap.get(method.getDeclaringClass()); if (methods != null && methods.contains(method)) { Object result=interceptor.intercept(new Invocation(target, method, args)); return result; } return method.invoke(target, args); } catch (Exception e) { throw ExceptionUtil.unwrapThrowable(e); }}

如前所述,插件的工作原理是基于責(zé)任鏈模式,可以注冊(cè)多個(gè)過濾器,層層包裝,最終由內(nèi)而外形成了一個(gè)近似裝飾器模式的責(zé)任鏈,最里面的基本實(shí)現(xiàn)是CachingExecutor:

淺談myBatis中的插件機(jī)制

從InterceptorChain的pluginAll方法可以看出這個(gè)結(jié)構(gòu)的構(gòu)造過程:

public Object pluginAll(Object target) { for (Interceptor interceptor : interceptors) { // 從這可以看出過濾器的傳遞的過程:動(dòng)態(tài)代理實(shí)例由內(nèi)而外層層包裝,類似于與裝飾器的結(jié)構(gòu),基礎(chǔ) 實(shí)現(xiàn)是一個(gè)Executor target = interceptor.plugin(target); } return target;}

這種由內(nèi)而外的包裝的棧結(jié)構(gòu)從外向內(nèi)層層代理調(diào)用,完成了責(zé)任鏈任務(wù)的逐級(jí)推送。從這個(gè)注冊(cè)過程可以看到,在list中越前面的Interceptor越先被代理,在棧結(jié)構(gòu)中越處于底層,執(zhí)行的順序越靠后。造成了注冊(cè)順序和執(zhí)行順序相反的現(xiàn)象。

插件的典型案例:PageHelper

pagehelper是一個(gè)實(shí)現(xiàn)物理分頁效果的開源插件,并且在底層通過Dialect類適配了不同的數(shù)據(jù)庫,其主要作用是攔截sql查詢,構(gòu)造一個(gè)查詢總數(shù)的新的以'_COUNT'結(jié)尾的新sql,最終再進(jìn)行分頁查詢。

自定義插件

定義Interceptor接口的實(shí)現(xiàn)類并在其上使用@Intercepts和@Signature注解進(jìn)行過濾的類和方法,比如定義一個(gè)打日志的插件:

@Intercepts({@Signature(type = Executor.class, method = 'query', args = { MappedStatement.class, Object.class,RowBounds.class, ResultHandler.class }),@Signature(type = Executor.class, method = 'query', args = { MappedStatement.class, Object.class,RowBounds.class, ResultHandler.class, CacheKey.class, BoundSql.class }), })public class LogInterceptor implements Interceptor {@Overridepublic Object intercept(Invocation invocation) throws Throwable {System.out.println('進(jìn)入了自定義的插件過濾器!');System.out.println('執(zhí)行的目標(biāo)是:' + invocation.getTarget());System.out.println('執(zhí)行的方法是:' + invocation.getMethod());System.out.println('執(zhí)行的參數(shù)是:' + invocation.getArgs());return invocation.proceed();}}

@Intercepts注解中包含了一個(gè)方法簽名數(shù)組,即@Signature數(shù)組,@Signature有三個(gè)屬性,type、method和args分別定義要攔截的類、方法名和參數(shù),這樣就可以通過反射唯一的確定了要攔截的方法。type即為在工作原理分析中提到的Executor、ParameterHandler、ResultSetHandler和StatementHandler,method配置對(duì)應(yīng)接口中的方法。

到此這篇關(guān)于淺談myBatis中的插件機(jī)制的文章就介紹到這了,更多相關(guān)myBatis 插件機(jī)制內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Mybatis 數(shù)據(jù)庫
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
av资源中文在线| 久久久久中文| 午夜国产精品视频免费体验区| 日本欧美不卡| 久久精品国产在热久久| 国产精品xvideos88| 欧美激情日韩| 精品久久91| 成人精品国产亚洲| av高清不卡| 欧美va天堂| 亚洲免费黄色| 久久亚洲色图| 日本精品国产| 欧美国产三级| 国产不卡人人| 欧美二区视频| 蜜臀久久99精品久久一区二区 | 在线日韩av| 欧美国产专区| 国产香蕉精品| 国产精品xxx| 精品国产欧美日韩| 麻豆成人在线观看| 国产精品国产一区| 先锋影音国产一区| 亚洲精品字幕| 日韩在线一区二区| 日本va欧美va瓶| 亚洲欧美日韩精品一区二区| 美女性感视频久久| 午夜在线精品偷拍| 蜜桃精品视频| 国产精品毛片一区二区三区| 日韩国产激情| 欧美伊人久久| 日韩av一二三| 国产欧美久久一区二区三区| 亚洲国产不卡| 成人日韩av| 美女一区网站| 亚洲伦乱视频| 国产二区精品| 久久久精品久久久久久96| 欧美日韩激情在线一区二区三区| 国产精品美女久久久浪潮软件| 麻豆视频观看网址久久| 丝袜美腿亚洲一区二区图片| 亚洲欧美成人综合| 日韩精品免费一区二区夜夜嗨| 在线亚洲国产精品网站| 久久久噜噜噜| 欧美一区二区三区激情视频| 日韩中文字幕一区二区三区| 日韩精品亚洲专区| 免费观看日韩电影| 亚洲欧洲免费| 国产精品伊人| 国产精品黑丝在线播放| 热三久草你在线| 欧美日韩精品免费观看视欧美高清免费大片| 香蕉精品视频在线观看| 国产一区二区三区不卡av | 中文国产一区| 中文字幕一区二区三区日韩精品 | 欧美日韩国产高清电影| 国产精品毛片aⅴ一区二区三区| 久久中文字幕二区| 国产精品白丝一区二区三区| 色一区二区三区| 91嫩草精品| 尤物在线精品| 国产精品原创| 欧美亚洲tv| 精品视频一区二区三区四区五区| 欧美在线亚洲综合一区| 日本va欧美va瓶| 欧美日韩国产高清电影| 久久精品一区| 亚洲人成网77777色在线播放 | 国产麻豆精品| 视频一区二区三区入口| 亚洲伦乱视频| 久久精品72免费观看| 视频在线观看一区二区三区| 999精品在线| 国产66精品| 欧美激情一区| 91成人精品在线| 免费成人性网站| 欧美日中文字幕| 国产情侣一区| 国产不卡一区| 亚洲人亚洲人色久| 成人看片网站| 91精品国产自产在线丝袜啪| 午夜精品网站| 日本国产精品| 国产视频一区二| 日韩精品久久久久久| 亚洲一区欧美激情| 激情婷婷综合| 国产99精品| 99久久婷婷这里只有精品| 一区二区精品伦理...| 成人污污视频| 精品视频一区二区三区四区五区 | 婷婷综合网站| 成人久久久久| 韩国精品主播一区二区在线观看| 久久亚州av| 欧美经典一区| 免费看一区二区三区| 国产毛片精品| 久久99蜜桃| 久久久久九九精品影院| 91九色综合| 国产精品一级| 国产精品草草| 福利一区视频| 成人影视亚洲图片在线| 国产成人精品福利| 国产图片一区| 国产精品毛片aⅴ一区二区三区| 久久激情五月激情| 国产亚洲欧美日韩在线观看一区二区| 日本国产欧美| 国产精品22p| 国产一区不卡| 日韩伦理一区| 99久久精品国产亚洲精品| 免费黄色成人| 性色一区二区| 婷婷视频一区二区三区| 日韩欧美三区| 国产乱码精品一区二区亚洲| 麻豆久久一区| 国产一区二区色噜噜| 香蕉成人av| 国产一区91| 欧美日韩1区| 国产精品99久久精品| 激情欧美丁香| 一区二区国产在线观看| 91精品一区| 国产一区丝袜| 国产专区一区| 蜜臀av一区二区三区| 日本免费在线视频不卡一不卡二| 国产欧美日韩影院| 色黄视频在线观看| 性欧美精品高清| 日本视频一区二区| 久久精品资源| 五月天久久777| 亚洲18在线| 精品亚洲成人| 亚洲播播91| 一区二区日韩免费看| 久久久久久亚洲精品美女| 成人片免费看| 亚洲欧洲av| 精品一区在线| 日韩精品视频中文字幕| 国产精品久久国产愉拍| 欧美日韩在线观看首页| 香蕉久久久久久久av网站| 中文字幕av亚洲精品一部二部| 欧美一区二区三区久久| 日韩av免费| 色综合视频一区二区三区日韩 | 日韩专区在线视频| 成人高清一区| 日韩不卡免费高清视频| 国产精品2区| 国产99精品一区| 91精品精品| 日本va欧美va欧美va精品| 视频在线不卡免费观看| 免费日韩av片| 美女精品视频在线| 亚洲女同一区| 国产精品任我爽爆在线播放| 成人久久久久| 麻豆国产欧美一区二区三区| 综合色一区二区| 亚洲久久在线| 国产日韩在线观看视频| 欧美黄色一区| 超碰99在线| 中文字幕av一区二区三区四区| 中文字幕免费一区二区| 国产乱码精品一区二区三区四区 | 视频一区日韩精品| 欧美激情视频一区二区三区免费 | 国产精品igao视频网网址不卡日韩| 蜜臀久久久99精品久久久久久| 欧美日韩高清| 免费不卡在线视频| 国模大尺度视频一区二区|