日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

java delegate怎么写_美团面试官:你说你们公司的Mybatis分页插件是你写的,给我说说它的设计原理?...

發(fā)布時間:2023/12/10 编程问答 38 豆豆
生活随笔 收集整理的這篇文章主要介紹了 java delegate怎么写_美团面试官:你说你们公司的Mybatis分页插件是你写的,给我说说它的设计原理?... 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

來源:http://my.oschina.net/zudajun

大多數(shù)框架,都支持插件,用戶可通過編寫插件來自行擴(kuò)展功能,Mybatis也不例外。

我們從插件配置、插件編寫、插件運(yùn)行原理、插件注冊與執(zhí)行攔截的時機(jī)、初始化插件、分頁插件的原理等六個方面展開闡述。

# 插件配置

Mybatis的插件配置在configuration內(nèi)部,初始化時,會讀取這些插件,保存于Configuration對象的InterceptorChain中。

<?xml version="1.0" encoding="UTF-8"?> public class Configuration { protected final InterceptorChain interceptorChain = new InterceptorChain();}

org.apache.ibatis.plugin.InterceptorChain.java源碼。

public class InterceptorChain { private final List interceptors = new ArrayList(); public Object pluginAll(Object target) { for (Interceptor interceptor : interceptors) { target = interceptor.plugin(target); } return target; } public void addInterceptor(Interceptor interceptor) { interceptors.add(interceptor); } public List getInterceptors() { return Collections.unmodifiableList(interceptors); }}

上面的for循環(huán)代表了只要是插件,都會以責(zé)任鏈的方式逐一執(zhí)行(別指望它能跳過某個節(jié)點(diǎn)),所謂插件,其實(shí)就類似于攔截器。

# 如何編寫一個插件

插件必須實(shí)現(xiàn)org.apache.ibatis.plugin.Interceptor接口。

public interface Interceptor { Object intercept(Invocation invocation) throws Throwable; Object plugin(Object target); void setProperties(Properties properties);}

intercept()方法:執(zhí)行攔截內(nèi)容的地方,比如想收點(diǎn)保護(hù)費(fèi)。由plugin()方法觸發(fā),interceptor.plugin(target)足以證明。

plugin()方法:決定是否觸發(fā)intercept()方法。

setProperties()方法:給自定義的攔截器傳遞xml配置的屬性參數(shù)。

下面自定義一個攔截器:

@Intercepts({ @Signature(type = Executor.class, method = "query", args = { MappedStatement.class, Object.class, RowBounds.class, ResultHandler.class }), @Signature(type = Executor.class, method = "close", args = { boolean.class }) })public class MyBatisInterceptor implements Interceptor { private Integer value; @Override public Object intercept(Invocation invocation) throws Throwable { return invocation.proceed(); } @Override public Object plugin(Object target) { System.out.println(value); // Plugin類是插件的核心類,用于給target創(chuàng)建一個JDK的動態(tài)代理對象,觸發(fā)intercept()方法 return Plugin.wrap(target, this); } @Override public void setProperties(Properties properties) { value = Integer.valueOf((String) properties.get("value")); }}

面對上面的代碼,我們需要解決兩個疑問:

1. ?為什么要寫Annotation注解?注解都是什么含義?

答:Mybatis規(guī)定插件必須編寫Annotation注解,是必須,而不是可選。

@Intercepts注解:裝載一個@Signature列表,一個@Signature其實(shí)就是一個需要攔截的方法封裝。那么,一個攔截器要攔截多個方法,自然就是一個@Signature列表。

type = Executor.class, method = "query", args = { MappedStatement.class, Object.class,?RowBounds.class, ResultHandler.class }

解釋:要攔截Executor接口內(nèi)的query()方法,參數(shù)類型為args列表。

2. Plugin.wrap(target, this)是干什么的?

答:使用JDK的動態(tài)代理,給target對象創(chuàng)建一個delegate代理對象,以此來實(shí)現(xiàn)方法攔截和增強(qiáng)功能,它會回調(diào)intercept()方法。

org.apache.ibatis.plugin.Plugin.java源碼:

public class Plugin implements InvocationHandler { private Object target; private Interceptor interceptor; private Map, Set> signatureMap; private Plugin(Object target, Interceptor interceptor, Map, Set> signatureMap) { this.target = target; this.interceptor = interceptor; this.signatureMap = signatureMap; } public static Object wrap(Object target, Interceptor interceptor) { Map, Set> signatureMap = getSignatureMap(interceptor); Class> type = target.getClass(); Class>[] interfaces = getAllInterfaces(type, signatureMap); if (interfaces.length > 0) { // 創(chuàng)建JDK動態(tài)代理對象 return Proxy.newProxyInstance( type.getClassLoader(), interfaces, new Plugin(target, interceptor, signatureMap)); } return target; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { try { Set methods = signatureMap.get(method.getDeclaringClass()); // 判斷是否是需要攔截的方法(很重要) if (methods != null && methods.contains(method)) { // 回調(diào)intercept()方法 return interceptor.intercept(new Invocation(target, method, args)); } return method.invoke(target, args); } catch (Exception e) { throw ExceptionUtil.unwrapThrowable(e); } }//...}

Map, Set> signatureMap:緩存需攔截對象的反射結(jié)果,避免多次反射,即target的反射結(jié)果。

所以,我們不要動不動就說反射性能很差,那是因?yàn)槟銢]有像Mybatis一樣去緩存一個對象的反射結(jié)果。

判斷是否是需要攔截的方法,這句注釋很重要,一旦忽略了,都不知道Mybatis是怎么判斷是否執(zhí)行攔截內(nèi)容的,要記住。

# Mybatis可以攔截哪些接口對象?

public class Configuration {//...public ParameterHandler newParameterHandler(MappedStatement mappedStatement, Object parameterObject, BoundSql boundSql) { ParameterHandler parameterHandler = mappedStatement.getLang().createParameterHandler(mappedStatement, parameterObject, boundSql); parameterHandler = (ParameterHandler) interceptorChain.pluginAll(parameterHandler); // 1 return parameterHandler; } public ResultSetHandler newResultSetHandler(Executor executor, MappedStatement mappedStatement, RowBounds rowBounds, ParameterHandler parameterHandler, ResultHandler resultHandler, BoundSql boundSql) { ResultSetHandler resultSetHandler = new DefaultResultSetHandler(executor, mappedStatement, parameterHandler, resultHandler, boundSql, rowBounds); resultSetHandler = (ResultSetHandler) interceptorChain.pluginAll(resultSetHandler); // 2 return resultSetHandler; } public StatementHandler newStatementHandler(Executor executor, MappedStatement mappedStatement, Object parameterObject, RowBounds rowBounds, ResultHandler resultHandler, BoundSql boundSql) { StatementHandler statementHandler = new RoutingStatementHandler(executor, mappedStatement, parameterObject, rowBounds, resultHandler, boundSql); statementHandler = (StatementHandler) interceptorChain.pluginAll(statementHandler); // 3 return statementHandler; } public Executor newExecutor(Transaction transaction) { return newExecutor(transaction, defaultExecutorType); } public Executor newExecutor(Transaction transaction, ExecutorType executorType) { executorType = executorType == null ? defaultExecutorType : executorType; executorType = executorType == null ? ExecutorType.SIMPLE : executorType; Executor executor; if (ExecutorType.BATCH == executorType) { executor = new BatchExecutor(this, transaction); } else if (ExecutorType.REUSE == executorType) { executor = new ReuseExecutor(this, transaction); } else { executor = new SimpleExecutor(this, transaction); } if (cacheEnabled) { executor = new CachingExecutor(executor); } executor = (Executor) interceptorChain.pluginAll(executor); // 4 return executor; }//...}

Mybatis只能攔截ParameterHandler、ResultSetHandler、StatementHandler、Executor共4個接口對象內(nèi)的方法。

重新審視interceptorChain.pluginAll()方法:該方法在創(chuàng)建上述4個接口對象時調(diào)用,其含義為給這些接口對象注冊攔截器功能,注意是注冊,而不是執(zhí)行攔截。

攔截器執(zhí)行時機(jī):plugin()方法注冊攔截器后,那么,在執(zhí)行上述4個接口對象內(nèi)的具體方法時,就會自動觸發(fā)攔截器的執(zhí)行,也就是插件的執(zhí)行。

所以,一定要分清,何時注冊,何時執(zhí)行。切不可認(rèn)為pluginAll()或plugin()就是執(zhí)行,它只是注冊。

#?Invocation

public class Invocation { private Object target; private Method method; private Object[] args;}

intercept(Invocation invocation)方法的參數(shù)Invocation ,我相信你一定可以看得懂,不解釋。

# 初始化插件源碼解析

org.apache.ibatis.builder.xml.XMLConfigBuilder.parseConfiguration(XNode)方法部分源碼。

pluginElement(root.evalNode("plugins")); 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).newInstance(); // 這里展示了setProperties()方法的調(diào)用時機(jī) interceptorInstance.setProperties(properties); configuration.addInterceptor(interceptorInstance); } } }

對于Mybatis,它并不區(qū)分是何種攔截器接口,所有的插件都是Interceptor,Mybatis完全依靠Annotation去標(biāo)識對誰進(jìn)行攔截,所以,具備接口一致性。

# 分頁插件原理

由于Mybatis采用的是邏輯分頁,而非物理分頁,那么,市場上就出現(xiàn)了可以實(shí)現(xiàn)物理分頁的Mybatis的分頁插件。

要實(shí)現(xiàn)物理分頁,就需要對String sql進(jìn)行攔截并增強(qiáng),Mybatis通過BoundSql對象存儲String sql,而BoundSql則由StatementHandler對象獲取。

public interface StatementHandler { List query(Statement statement, ResultHandler resultHandler) throws SQLException; BoundSql getBoundSql();}public class BoundSql { public String getSql() { return sql; }}

因此,就需要編寫一個針對StatementHandler的query方法攔截器,然后獲取到sql,對sql進(jìn)行重寫增強(qiáng)。

任它天高海闊,任它變化無窮,我們只要懂得原理,再多插件,我們都可以對其投送王之蔑視。

熱文推薦

Java面試題匯總(04)

12月DB-Engines數(shù)據(jù)庫排名,你猜誰會是第一?

老弟,你連HTTPS 原理都不懂,還給我講“中間人攻擊”,逗我嗎...


覺得不錯,請給個「在看」

分享給你的朋友!

- End -

創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎

總結(jié)

以上是生活随笔為你收集整理的java delegate怎么写_美团面试官:你说你们公司的Mybatis分页插件是你写的,给我说说它的设计原理?...的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。

主站蜘蛛池模板: 国产网站黄 | 国产农村妇女毛片精品 | 亚洲毛片大全 | 97在线免费观看 | 国产人妻777人伦精品hd | 日本免费三片在线播放 | 手机看片亚洲 | 国际av在线 | 在线观看免费小视频 | 在线国产区| 日本xx视频 | 欧美在线www | 青春草国产视频 | 本道综合精品 | 成人天堂av | 麻豆黄色片 | 永久免费AV无码网站韩国毛片 | 大陆极品少妇内射aaaaaa | 男生插女生视频在线观看 | 国产又爽又黄免费软件 | 日本免费一区二区三区四区 | 欧洲xxxxx| 成人欧美在线视频 | 激情综合图区 | 三级欧美韩日大片在线看 | 黄色伊人网 | 国产精品久久久不卡 | 日韩在线播放视频 | 国产一毛片 | www青青草 | 成人三级电影网站 | 香港台湾日本三级大全 | 四虎最新站名点击进入 | 久热免费在线视频 | 天堂中文在线视频 | 天天操天天操天天操天天 | 精品久久人人妻人人做人人 | 日韩在线观看第一页 | 高清不卡一区二区 | 国产又粗又猛又爽又黄的视频小说 | 超碰女| 亚洲第一男人天堂 | 欧美高清日韩 | 狠狠干伊人网 | 国产伦精品视频一区二区三区 | 国产真实生活伦对白 | 日韩免费一区二区三区 | 2025韩国大尺度电影 | 国产精品久久9 | 日本老熟妇毛茸茸 | 蜜桃视频一区二区三区在线观看 | 亚洲精品一区二三区不卡 | 欧美日韩精品 | 日韩精品一区二区三区在线视频 | 国产精品美女高潮无套 | 奴性白洁会所调教 | 亚洲一级片网站 | 91人妻一区二区三区 | 日本爱爱免费视频 | 北条麻妃二三区 | 超碰人人草 | 九九热视频在线播放 | 激情五月色播五月 | 日本久久一级片 | 夜夜撸av | 淫岳高潮记小说 | 91传媒视频在线观看 | 欧美整片sss | 少女情窦初开的第4集在线观看 | 欧美伦理在线观看 | 亚洲乱论 | 天天摸天天操 | 香港三级网站 | 久久精品男人的天堂 | yy4138理论片动漫理论片 | 欧美成人久久久免费播放 | 日本视频在线观看 | 日韩城人视频 | 丝袜国产视频 | 国产福利91精品 | 久草99 | 日本一区二区三区四区在线观看 | 欧美专区第一页 | 欧美日韩一区二区三区在线 | 久久精品久久精品久久 | 男人的天堂在线播放 | 午夜毛片 | 青青操在线观看 | 亚洲精品视频播放 | 久久嗨| 婷婷四房综合激情五月 | 椎名由奈在线观看 | 日本丰满肉感bbwbbwbbw | 麻豆激情视频 | 香蕉尹人网 | 国产原创精品 | 爱爱15p| 天天综合天天综合 | 亚洲第一免费 |