mybatis中的擴(kuò)展實(shí)現(xiàn)源碼解析
前言
最近項(xiàng)目中需要用到mybatis的擴(kuò)展,就深入看了下mybatis的實(shí)現(xiàn),對(duì)其靈活性和擴(kuò)展性的設(shè)計(jì)思想還是非常佩服的
首先說(shuō)一下mybatis的攔截器使用方法:繼承其Intercepter接口,實(shí)現(xiàn)org.apache.ibatis.plugin.Interceptor#intercept
方法,在其中或者對(duì)其要執(zhí)行的方法進(jìn)行攔截,或者對(duì)返回值進(jìn)行解析
同時(shí)基于org.apache.ibatis.plugin.Intercepts
和org.apache.ibatis.plugin.Signature
這兩個(gè)注解來(lái)決定,對(duì)哪些執(zhí)行器的哪些方法進(jìn)行攔截
先看下攔截器的核心接口
public interface Interceptor { Object intercept(Invocation invocation) throws Throwable; Object plugin(Object target); void setProperties(Properties properties); }
其中intercept方法是核心方法,攔截器的實(shí)現(xiàn),plugin方法是用于配置哪些對(duì)哪些執(zhí)行器進(jìn)行攔截
繼續(xù)看源碼,可以看到mybatis的攔截是使用了jdk的動(dòng)態(tài)代理實(shí)現(xiàn)的,本質(zhì)上是一種代理機(jī)制
public class Plugin implements InvocationHandler { private final Object target; private final Interceptor interceptor; private final Map<Class<?>, Set<Method>> signatureMap; private Plugin(Object target, Interceptor interceptor, Map<Class<?>, Set<Method>> signatureMap) { this.target = target; this.interceptor = interceptor; this.signatureMap = signatureMap; } public static Object wrap(Object target, Interceptor interceptor) { Map<Class<?>, Set<Method>> signatureMap = getSignatureMap(interceptor); Class<?> type = target.getClass(); Class<?>[] interfaces = getAllInterfaces(type, signatureMap); if (interfaces.length > 0) { 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<Method> methods = signatureMap.get(method.getDeclaringClass()); if (methods != null && methods.contains(method)) { return interceptor.intercept(new Invocation(target, method, args)); } return method.invoke(target, args); } catch (Exception e) { throw ExceptionUtil.unwrapThrowable(e); } } ...... }
mybatis的這個(gè)Plugin就是代理類,這個(gè)代理類是在org.apache.ibatis.plugin.Interceptor#plugin
方法中初始化的(調(diào)用org.apache.ibatis.plugin.Plugin#wrap
),一個(gè)Plugin包含一個(gè)Intercepter,以及該Intercepter相關(guān)的注解配置信息,當(dāng)對(duì)攔截對(duì)象的對(duì)應(yīng)方法進(jìn)行執(zhí)行的時(shí)候,都會(huì)根據(jù)這些注解配置來(lái)判斷是否需要執(zhí)行該代理攔截(org.apache.ibatis.plugin.Plugin#invoke
)
再看下plugin是如何被加載的:
public class InterceptorChain { private final List<Interceptor> interceptors = new ArrayList<Interceptor>(); 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<Interceptor> getInterceptors() { return Collections.unmodifiableList(interceptors); } }
org.apache.ibatis.plugin.Interceptor#plugin
是在org.apache.ibatis.plugin.InterceptorChain#pluginAll
方法中調(diào)用的,我們可以看到,如果一個(gè)應(yīng)用中注冊(cè)了多個(gè)攔截器,那么實(shí)際上是會(huì)進(jìn)行一個(gè)for循環(huán)的加載,由于上面說(shuō)到了,加載一次,本質(zhì)上是對(duì)mybatis的執(zhí)行期進(jìn)行一次代理包裝,那么加載多次的話,就會(huì)代理包裝多次,實(shí)際上就是一種多重代理了,這樣就保證了每次調(diào)用都會(huì)按照代理順序進(jìn)行調(diào)用和返回的處理
可以看到,在做這些mybatis執(zhí)行器初始化的時(shí)候,都會(huì)進(jìn)行攔截器鏈的加載
至此,mybatis基于jdk動(dòng)態(tài)代理的擴(kuò)展實(shí)現(xiàn)方法就了解清楚了,其靈活性在于,它抽象了執(zhí)行器的概念,并且攔截器的攔截方法也是固定的,我們可以對(duì)不同執(zhí)行器的不同方法進(jìn)行攔截,而對(duì)這些擴(kuò)展點(diǎn)進(jìn)行擴(kuò)展卻不用寫多個(gè)方法實(shí)現(xiàn)多個(gè)方法,只需要實(shí)現(xiàn)一個(gè)接口就可以搞定了!
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)腳本之家的支持。
相關(guān)文章
Java BufferedImage轉(zhuǎn)換為MultipartFile方式
這篇文章主要介紹了Java BufferedImage轉(zhuǎn)換為MultipartFile方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-09-09SpringBoot使用Thymeleaf自定義標(biāo)簽的實(shí)例代碼
這篇文章主要介紹了SpringBoot使用Thymeleaf自定義標(biāo)簽的實(shí)例代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-09-09Java Socket實(shí)現(xiàn)多線程通信功能示例
這篇文章主要介紹了Java Socket實(shí)現(xiàn)多線程通信功能,結(jié)合具體實(shí)例形式較為詳細(xì)的分析了java多線程通信的原理及客戶端、服務(wù)器端相應(yīng)實(shí)現(xiàn)技巧,需要的朋友可以參考下2017-06-06Java通過(guò)Scanner了解if...else if語(yǔ)句
這篇文章主要介紹了Java通過(guò)Scanner了解if...else if語(yǔ)句,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-01-01java中對(duì)字符串每個(gè)字符統(tǒng)計(jì)的方法
java中對(duì)字符串每個(gè)字符統(tǒng)計(jì)的方法,需要的朋友可以參考一下2013-03-03Java函數(shù)式編程(三):列表的轉(zhuǎn)化
這篇文章主要介紹了Java函數(shù)式編程(二):列表的轉(zhuǎn)化,lambda表達(dá)式不僅能幫助我們遍歷集合,并且可以進(jìn)行集合的轉(zhuǎn)化,需要的朋友可以參考下2014-09-09Windows下java、javaw、javaws以及jvm.dll等進(jìn)程的區(qū)別
這篇文章主要介紹了Windows下java、javaw、javaws以及jvm.dll等進(jìn)程的區(qū)別,本文分別講解了它們的作用并給出代碼實(shí)例,最后做出了區(qū)別總結(jié),需要的朋友可以參考下2015-03-03MybatisPlus中@TableField注解的使用詳解
這篇文章主要介紹了MybatisPlus中@TableField注解的使用詳解,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09