前言
最近項目中需要用到mybatis的擴展,就深入看了下mybatis的實現,對其靈活性和擴展性的設計思想還是非常佩服的
首先說一下mybatis的攔截器使用方法:繼承其intercepter接口,實現org.apache.ibatis.plugin.interceptor#intercept
方法,在其中或者對其要執行的方法進行攔截,或者對返回值進行解析
同時基于org.apache.ibatis.plugin.intercepts
和org.apache.ibatis.plugin.signature
這兩個注解來決定,對哪些執行器的哪些方法進行攔截
先看下攔截器的核心接口
1
2
3
4
5
6
7
8
9
|
public interface interceptor { object intercept(invocation invocation) throws throwable; object plugin(object target); void setproperties(properties properties); } |
其中intercept方法是核心方法,攔截器的實現,plugin方法是用于配置哪些對哪些執行器進行攔截
繼續看源碼,可以看到mybatis的攔截是使用了jdk的動態代理實現的,本質上是一種代理機制
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
|
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的這個plugin就是代理類,這個代理類是在org.apache.ibatis.plugin.interceptor#plugin
方法中初始化的(調用org.apache.ibatis.plugin.plugin#wrap
),一個plugin包含一個intercepter,以及該intercepter相關的注解配置信息,當對攔截對象的對應方法進行執行的時候,都會根據這些注解配置來判斷是否需要執行該代理攔截(org.apache.ibatis.plugin.plugin#invoke
)
再看下plugin是如何被加載的:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
|
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
方法中調用的,我們可以看到,如果一個應用中注冊了多個攔截器,那么實際上是會進行一個for循環的加載,由于上面說到了,加載一次,本質上是對mybatis的執行期進行一次代理包裝,那么加載多次的話,就會代理包裝多次,實際上就是一種多重代理了,這樣就保證了每次調用都會按照代理順序進行調用和返回的處理
可以看到,在做這些mybatis執行器初始化的時候,都會進行攔截器鏈的加載
至此,mybatis基于jdk動態代理的擴展實現方法就了解清楚了,其靈活性在于,它抽象了執行器的概念,并且攔截器的攔截方法也是固定的,我們可以對不同執行器的不同方法進行攔截,而對這些擴展點進行擴展卻不用寫多個方法實現多個方法,只需要實現一個接口就可以搞定了!
總結
以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對服務器之家的支持。
原文鏈接:http://www.cnblogs.com/cishengchongyan/p/10208040.html