电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術文章
文章詳情頁

淺談myBatis中的插件機制

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

插件的配置與使用

在mybatis-config.xml配置文件中配置plugin結點,比如配置一個自定義的日志插件LogInterceptor和一個開源的分頁插件PageInterceptor:

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

插件的工作原理

借助責任鏈模式,定義一系列的過濾器,在查詢等方法執行時進行過濾,從而達到控制參數、調整查詢語句和控制查詢結果等作用。下面從插件的加載(初始化)、注冊和調用這三個方面闡述插件的工作原理。

過濾器的加載(初始化)

和其他配置信息一樣,過濾器的加載也會在myBatis讀取配置文件創建Configuration對象時進行,相應的信息存儲在Configuration的interceptorChain屬性中,InterceptorChain封裝了一個包含Interceptor的list:

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

在XMLConfigBuilder進行解析配置文件時執行pluginElement方法,生成過濾器實例,并添加到上述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); } }}

過濾器的注冊

可以為Executor、ParameterHandler、ResultSetHandler和StatementHandler四個接口注冊過濾器,注冊的時機也就是這四種接口的實現類的對象的生成時機,比如Executor的過濾器的注冊發生在SqlSessionFactory使用openSession方法構建SqlSession的過程中(因為SqlSession依賴一個Executor實例),ParameterHandler和StatementHandler的過濾器發生在doQuery等sql執行方法執行時注冊,而ResultHandler的過濾器的注冊則發生在查詢結果返回給客戶端的過程中。以Executor的過濾器的注冊為例,經過了這樣的過程:

淺談myBatis中的插件機制

現在詳細的分析一下Plugin的wrap這個靜態的包裝方法:

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

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

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方法是依據不同的目標對象(Executor等四種)進行過濾的過程,只給對應的目標進行注冊:

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()]);}

至此,實際使用的Executor對象將是通過動態代理生成的Plugin實例。

過濾器的調用

在第二步中完成了過濾器的注冊,在實際調用Executor時,將由實現了InvocationHandler接口的Plugin實例進行接管,對Executor相應方法方法的調用,將實際上調用動態代理體系下的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); }}

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

淺談myBatis中的插件機制

從InterceptorChain的pluginAll方法可以看出這個結構的構造過程:

public Object pluginAll(Object target) { for (Interceptor interceptor : interceptors) { // 從這可以看出過濾器的傳遞的過程:動態代理實例由內而外層層包裝,類似于與裝飾器的結構,基礎 實現是一個Executor target = interceptor.plugin(target); } return target;}

這種由內而外的包裝的棧結構從外向內層層代理調用,完成了責任鏈任務的逐級推送。從這個注冊過程可以看到,在list中越前面的Interceptor越先被代理,在棧結構中越處于底層,執行的順序越靠后。造成了注冊順序和執行順序相反的現象。

插件的典型案例:PageHelper

pagehelper是一個實現物理分頁效果的開源插件,并且在底層通過Dialect類適配了不同的數據庫,其主要作用是攔截sql查詢,構造一個查詢總數的新的以'_COUNT'結尾的新sql,最終再進行分頁查詢。

自定義插件

定義Interceptor接口的實現類并在其上使用@Intercepts和@Signature注解進行過濾的類和方法,比如定義一個打日志的插件:

@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('進入了自定義的插件過濾器!');System.out.println('執行的目標是:' + invocation.getTarget());System.out.println('執行的方法是:' + invocation.getMethod());System.out.println('執行的參數是:' + invocation.getArgs());return invocation.proceed();}}

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

到此這篇關于淺談myBatis中的插件機制的文章就介紹到這了,更多相關myBatis 插件機制內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Mybatis 數據庫
相關文章:
主站蜘蛛池模板: 叉车电池-叉车电瓶-叉车蓄电池-铅酸蓄电池-电动叉车蓄电池生产厂家 | ASA膜,ASA共挤料,篷布色母料-青岛未来化学有限公司 | 超声骨密度仪-动脉硬化检测仪器-人体成分分析仪厂家/品牌/价格_南京科力悦 | 泰国试管婴儿_泰国第三代试管婴儿费用|成功率|医院—新生代海外医疗 | 胜为光纤光缆_光纤跳线_单模尾纤_光纤收发器_ODF光纤配线架厂家直销_北京睿创胜为科技有限公司 - 北京睿创胜为科技有限公司 | 海尔生物医疗四川代理商,海尔低温冰箱四川销售-成都壹科医疗器械有限公司 | 5nd音乐网|最新流行歌曲|MP3歌曲免费下载|好听的歌|音乐下载 免费听mp3音乐 | 广州展台特装搭建商|特装展位设计搭建|展会特装搭建|特装展台制作设计|展览特装公司 | 浙江栓钉_焊钉_剪力钉厂家批发_杭州八建五金制造有限公司 | 北京公积金代办/租房发票/租房备案-北京金鼎源公积金提取服务中心 | 天津电机维修|水泵维修-天津晟佳机电设备有限公司 | 动库网动库商城-体育用品专卖店:羽毛球,乒乓球拍,网球,户外装备,运动鞋,运动包,运动服饰专卖店-正品运动品网上商城动库商城网 - 动库商城 | 全自动烧卖机厂家_饺子机_烧麦机价格_小笼汤包机_宁波江北阜欣食品机械有限公司 | 合肥宠物店装修_合肥宠物美容院装修_合肥宠物医院设计装修公司-安徽盛世和居装饰 | 硬度计_影像测量仪_维氏硬度计_佛山市精测计量仪器设备有限公司厂家 | 北钻固控设备|石油钻采设备-石油固控设备厂家 | 金属清洗剂,防锈油,切削液,磨削液-青岛朗力防锈材料有限公司 | Safety light curtain|Belt Sway Switches|Pull Rope Switch|ultrasonic flaw detector-Shandong Zhuoxin Machinery Co., Ltd | 齿轮减速机_齿轮减速电机-VEMT蜗轮蜗杆减速机马达生产厂家瓦玛特传动瑞环机电 | 安徽合肥格力空调专卖店_格力中央空调_格力空调总经销公司代理-皖格制冷设备 | 刮板输送机,粉尘加湿搅拌机,螺旋输送机,布袋除尘器 | 宁波普瑞思邻苯二甲酸盐检测仪,ROHS2.0检测设备,ROHS2.0测试仪厂家 | pos机办理,智能/扫码/二维码/微信支付宝pos机-北京万汇通宝商贸有限公司 | 安徽成考网-安徽成人高考网 | 耐高温电缆厂家-远洋高温电缆| 青岛美佳乐清洁工程有限公司|青岛油烟管道清洗|酒店|企事业单位|学校工厂厨房|青岛油烟管道清洗 插针变压器-家用电器变压器-工业空调变压器-CD型电抗器-余姚市中驰电器有限公司 | 医院专用门厂家报价-医用病房门尺寸大全-抗菌木门品牌推荐 | 盛源真空泵|空压机-浙江盛源空压机制造有限公司-【盛源官网】 | 天然气分析仪-液化气二甲醚分析仪|传昊仪器 | 剪刃_纵剪机刀片_分条机刀片-南京雷德机械有限公司 | 济南货架定做_仓储货架生产厂_重型货架厂_仓库货架批发_济南启力仓储设备有限公司 | 艺术涂料_进口艺术涂料_艺术涂料加盟_艺术涂料十大品牌 -英国蒙太奇艺术涂料 | 交变/复合盐雾试验箱-高低温冲击试验箱_安奈设备产品供应杭州/江苏南京/安徽马鞍山合肥等全国各地 | 小港信息港-鹤壁信息港 鹤壁老百姓便民生活信息网站 | 医养体检包_公卫随访箱_慢病随访包_家签随访包_随访一体机-济南易享医疗科技有限公司 | 电缆接头-防爆电缆接头-格兰头-金属电缆接头-防爆填料函 | 煤棒机_增碳剂颗粒机_活性炭颗粒机_木炭粉成型机-巩义市老城振华机械厂 | PCB厂|线路板厂|深圳线路板厂|软硬结合板厂|电路板生产厂家|线路板|深圳电路板厂家|铝基板厂家|深联电路-专业生产PCB研发制造 | 精密交叉滚子轴承厂家,转盘轴承,YRT转台轴承-洛阳千协轴承 | ALC墙板_ALC轻质隔墙板_隔音防火墙板_轻质隔墙材料-湖北博悦佳 | 除湿机|工业除湿机|抽湿器|大型地下室车间仓库吊顶防爆除湿机|抽湿烘干房|新风除湿机|调温/降温除湿机|恒温恒湿机|加湿机-杭州川田电器有限公司 |