1 AOP各類的實現java
AOP就是面向切面編程,咱們能夠從幾個層面來實現AOP。編程
在編譯器修改源代碼,在運行期字節碼加載前修改字節碼或字節碼加載後動態建立代理類的字節碼,如下是各類實現機制的比較。 緩存
類別框架 |
機制ide |
原理函數 |
優勢性能 |
缺點測試 |
靜態AOPthis |
靜態織入lua |
在編譯期,切面直接以字節碼的形式編譯到目標字節碼文件中。 |
對系統無性能影響。 |
靈活性不夠。 |
動態AOP |
動態代理 |
在運行期,目標類加載後,爲接口動態生成代理類,將切面植入到代理類中。 |
相對於靜態AOP更加靈活。 |
切入的關注點須要實現接口。對系統有一點性能影響。 |
動態字節碼生成 |
在運行期,目標類加載後,動態構建字節碼文件生成目標類的子類,將切面邏輯加入到子類中。 |
沒有接口也能夠織入。 |
擴展類的實例方法爲final時,則沒法進行織入。 |
自定義類加載器 |
在運行期,目標加載前,將切面邏輯加到目標字節碼裏。 |
能夠對絕大部分類進行織入。 |
代碼中若是使用了其餘類加載器,則這些類將不會被織入。 |
字節碼轉換 |
在運行期,全部類加載器加載字節碼前,前進行攔截。 |
能夠對全部類進行織入。 |
|
2 AOP裏的公民
- Joinpoint:攔截點,如某個業務方法。
- Pointcut:Joinpoint的表達式,表示攔截哪些方法。一個Pointcut對應多個Joinpoint。
- Advice: 要切入的邏輯。
- Before Advice 在方法前切入。
- After Advice 在方法後切入,拋出異常時也會切入。
- After Returning Advice 在方法返回後切入,拋出異常則不會切入。
- After Throwing Advice 在方法拋出異常時切入。
- Around Advice 在方法執行先後切入,能夠中斷或忽略原有流程的執行。
- 公民之間的關係
織入器經過在切面中定義pointcut來搜索目標(被代理類)的JoinPoint(切入點),而後把要切入的邏輯(Advice)織入到目標對象裏,生成代理類。
3 AOP的實現機制
本章節將詳細介紹AOP有各類實現機制。
3.1 動態代理
Java在JDK1.3後引入的動態代理機制,使咱們能夠在運行期動態的建立代理類。使用動態代理實現AOP須要有四個角色:被代理的類,被代理類的接口,織入器,和InvocationHandler,而織入器使用接口反射機制生成一個代理類,而後在這個代理類中織入代碼。被代理的類是AOP裏所說的目標,InvocationHandler是切面,它包含了Advice和Pointcut。
3.1.1 使用動態代理
那如何使用動態代理來實現AOP。下面的例子演示在方法執行前織入一段記錄日誌的代碼,其中Business是代理類,LogInvocationHandler是記錄日誌的切面,IBusiness, IBusiness2是代理類的接口,Proxy.newProxyInstance是織入器。
清單一:動態代理的演示
- public static void main(String[] args) {
- //須要代理的接口,被代理類實現的多個接口都必須在這裏定義
- Class[] proxyInterface = new Class[] { IBusiness.class, IBusiness2.class };
- //構建AOP的Advice,這裏須要傳入業務類的實例
- LogInvocationHandler handler = new LogInvocationHandler(new Business());
- //生成代理類的字節碼加載器
- ClassLoader classLoader = DynamicProxyDemo.class.getClassLoader();
- //織入器,織入代碼並生成代理類
- IBusiness2 proxyBusiness = (IBusiness2) Proxy.newProxyInstance(classLoader, proxyInterface, handler);
- //使用代理類的實例來調用方法。
- proxyBusiness.doSomeThing2();
- ((IBusiness) proxyBusiness).doSomeThing();
- }
-
- /**
- * 打印日誌的切面
- */
- public static class LogInvocationHandler implements InvocationHandler {
-
- private Object target; //目標對象
-
- LogInvocationHandler(Object target) {
- this.target = target;
- }
-
- @Override
- public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
- //執行原有邏輯
- Object rev = method.invoke(target, args);
- //執行織入的日誌,你能夠控制哪些方法執行切入邏輯
- if (method.getName().equals("doSomeThing2")) {
- System.out.println("記錄日誌");
- }
- return rev;
- }
- }
-
- 接口IBusiness和IBusiness2定義省略。
業務類,須要代理的類。
- public class Business implements IBusiness, IBusiness2 {
-
- @Override
- public boolean doSomeThing() {
- System.out.println("執行業務邏輯");
- return true;
- }
-
- @Override
- public void doSomeThing2() {
- System.out.println("執行業務邏輯2");
- }
-
- }
輸出
能夠看到「記錄日誌」的邏輯切入到Business類的doSomeThing方法前了。
3.1.2 動態代理原理
本節將結合動態代理的源代碼講解其實現原理。動態代理的核心其實就是代理對象的生成,即Proxy.newProxyInstance(classLoader, proxyInterface, handler)。讓咱們進入newProxyInstance方法觀摩下,核心代碼其實就三行。
清單二:生成代理類
- //獲取代理類
- Class cl = getProxyClass(loader, interfaces);
- //獲取帶有InvocationHandler參數的構造方法
- Constructor cons = cl.getConstructor(constructorParams);
- //把handler傳入構造方法生成實例
- return (Object) cons.newInstance(new Object[] { h });
其中getProxyClass(loader, interfaces)方法用於獲取代理類,它主要作了三件事情:在當前類加載器的緩存裏搜索是否有代理類,沒有則生成代理類並緩存在本地JVM裏。
清單三:查找代理類。
- // 緩存的key使用接口名稱生成的List
- Object key = Arrays.asList(interfaceNames);
- synchronized (cache) {
- do {
- Object value = cache.get(key);
- // 緩存裏保存了代理類的引用
- if (value instanceof Reference) {
- proxyClass = (Class) ((Reference) value).get();
- }
- if (proxyClass != null) {
- // 代理類已經存在則返回
- return proxyClass;
- } else if (value == pendingGenerationMarker) {
- // 若是代理類正在產生,則等待
- try {
- cache.wait();
- } catch (InterruptedException e) {
- }
- continue;
- } else {
- //沒有代理類,則標記代理準備生成
- cache.put(key, pendingGenerationMarker);
- break;
- }
- } while (true);
- }
代理類的生成主要是如下這兩行代碼。
清單四:生成並加載代理類
- //生成代理類的字節碼文件並保存到硬盤中(默認不保存到硬盤)
- proxyClassFile = ProxyGenerator.generateProxyClass(proxyName, interfaces);
- //使用類加載器將字節碼加載到內存中
- proxyClass = defineClass0(loader, proxyName,proxyClassFile, 0, proxyClassFile.length);
ProxyGenerator.generateProxyClass()方法屬於sun.misc包下,Oracle並無提供源代碼,可是咱們可使用JD-GUI這樣的反編譯軟件打開jre\lib\rt.jar來一探究竟,如下是其核心代碼的分析。
清單五:代理類的生成過程
- //添加接口中定義的方法,此時方法體爲空
- for (int i = 0; i < this.interfaces.length; i++) {
- localObject1 = this.interfaces[i].getMethods();
- for (int k = 0; k < localObject1.length; k++) {
- addProxyMethod(localObject1[k], this.interfaces[i]);
- }
- }
-
- //添加一個帶有InvocationHandler的構造方法
- MethodInfo localMethodInfo = new MethodInfo("<init>", "(Ljava/lang/reflect/InvocationHandler;)V", 1);
-
- //循環生成方法體代碼(省略)
- //方法體裏生成調用InvocationHandler的invoke方法代碼。(此處有所省略)
- this.cp.getInterfaceMethodRef("InvocationHandler", "invoke", "Object; Method; Object;")
-
- //將生成的字節碼,寫入硬盤,前面有個if判斷,默認狀況下不保存到硬盤。
- localFileOutputStream = new FileOutputStream(ProxyGenerator.access$000(this.val$name) + ".class");
- localFileOutputStream.write(this.val$classFile);
那麼經過以上分析,咱們能夠推出動態代理爲咱們生成了一個這樣的代理類。把方法doSomeThing的方法體修改成調用LogInvocationHandler的invoke方法。
清單六:生成的代理類源碼
- public class ProxyBusiness implements IBusiness, IBusiness2 {
-
- private LogInvocationHandler h;
-
- @Override
- public void doSomeThing2() {
- try {
- Method m = (h.target).getClass().getMethod("doSomeThing", null);
- h.invoke(this, m, null);
- } catch (Throwable e) {
- // 異常處理(略)
- }
- }
-
- @Override
- public boolean doSomeThing() {
- try {
- Method m = (h.target).getClass().getMethod("doSomeThing2", null);
- return (Boolean) h.invoke(this, m, null);
- } catch (Throwable e) {
- // 異常處理(略)
- }
- return false;
- }
-
- public ProxyBusiness(LogInvocationHandler h) {
- this.h = h;
- }
-
- //測試用
- public static void main(String[] args) {
- //構建AOP的Advice
- LogInvocationHandler handler = new LogInvocationHandler(new Business());
- new ProxyBusiness(handler).doSomeThing();
- new ProxyBusiness(handler).doSomeThing2();
- }
- }
3.1.3 小結
從前兩節的分析咱們能夠看出,動態代理在運行期經過接口動態生成代理類,這爲其帶來了必定的靈活性,但這個靈活性卻帶來了兩個問題,第一代理類必須實現一個接口,若是沒實現接口會拋出一個異常。第二性能影響,由於動態代理使用反射的機制實現的,首先反射確定比直接調用要慢,通過測試大概每一個代理類比靜態代理多出10幾毫秒的消耗。其次使用反射大量生成類文件可能引發Full GC形成性能影響,由於字節碼文件加載後會存放在JVM運行時區的方法區(或者叫持久代)中,當方法區滿的時候,會引發Full GC,因此當你大量使用動態代理時,能夠將持久代設置大一些,減小Full GC次數。
3.2 動態字節碼生成
使用動態字節碼生成技術實現AOP原理是在運行期間目標字節碼加載後,生成目標類的子類,將切面邏輯加入到子類中,因此使用Cglib實現AOP不須要基於接口。
本節介紹如何使用Cglib來實現動態字節碼技術。Cglib是一個強大的,高性能的Code生成類庫,它能夠在運行期間擴展Java類和實現Java接口,它封裝了Asm,因此使用Cglib前須要引入Asm的jar。 清單七:使用CGLib實現AOP
- public static void main(String[] args) {
- byteCodeGe();
- }
-
- public static void byteCodeGe() {
- //建立一個織入器
- Enhancer enhancer = new Enhancer();
- //設置父類
- enhancer.setSuperclass(Business.class);
- //設置須要織入的邏輯
- enhancer.setCallback(new LogIntercept());
- //使用織入器建立子類
- IBusiness2 newBusiness = (IBusiness2) enhancer.create();
- newBusiness.doSomeThing2();
- }
-
- /**
- * 記錄日誌
- */
- public static class LogIntercept implements MethodInterceptor {
-
- @Override
- public Object intercept(Object target, Method method, Object[] args, MethodProxy proxy) throws Throwable {
- //執行原有邏輯,注意這裏是invokeSuper
- Object rev = proxy.invokeSuper(target, args);
- //執行織入的日誌
- if (method.getName().equals("doSomeThing2")) {
- System.out.println("記錄日誌");
- }
- return rev;
- }
- }
3.3 自定義類加載器
若是咱們實現了一個自定義類加載器,在類加載到JVM以前直接修改某些類的方法,並將切入邏輯織入到這個方法裏,而後將修改後的字節碼文件交給虛擬機運行,那豈不是更直接。
Javassist是一個編輯字節碼的框架,可讓你很簡單地操做字節碼。它能夠在運行期定義或修改Class。使用Javassist實現AOP的原理是在字節碼加載前直接修改須要切入的方法。這比使用Cglib實現AOP更加高效,而且沒太多限制,實現原理以下圖:
咱們使用系統類加載器啓動咱們自定義的類加載器,在這個類加載器里加一個類加載監聽器,監聽器發現目標類被加載時就織入切入邏輯,我們再看看使用Javassist實現AOP的代碼:
清單八:啓動自定義的類加載器
- //獲取存放CtClass的容器ClassPool
- ClassPool cp = ClassPool.getDefault();
- //建立一個類加載器
- Loader cl = new Loader();
- //增長一個轉換器
- cl.addTranslator(cp, new MyTranslator());
- //啓動MyTranslator的main函數
- cl.run("jsvassist.JavassistAopDemo$MyTranslator", args);
清單九:類加載監聽器
- public static class MyTranslator implements Translator {
-
- public void start(ClassPool pool) throws NotFoundException, CannotCompileException {
- }
-
- /* *
- * 類裝載到JVM前進行代碼織入
- */
- public void onLoad(ClassPool pool, String classname) {
- if (!"model$Business".equals(classname)) {
- return;
- }
- //經過獲取類文件
- try {
- CtClass cc = pool.get(classname);
- //得到指定方法名的方法
- CtMethod m = cc.getDeclaredMethod("doSomeThing");
- //在方法執行前插入代碼
- m.insertBefore("{ System.out.println(\"記錄日誌\"); }");
- } catch (NotFoundException e) {
- } catch (CannotCompileException e) {
- }
- }
-
- public static void main(String[] args) {
- Business b = new Business();
- b.doSomeThing2();
- b.doSomeThing();
- }
- }
輸出:
其中Bussiness類在本文的清單一中定義。看起來是否是特別簡單,CtClass是一個class文件的抽象描述。我們也可使用insertAfter()在方法的末尾插入代碼,使用insertAt()在指定行插入代碼。
3.3.1 小結
從本節中可知,使用自定義的類加載器實現AOP在性能上要優於動態代理和Cglib,由於它不會產生新類,可是它仍然存在一個問題,就是若是其餘的類加載器來加載類的話,這些類將不會被攔截。
3.4 字節碼轉換
自定義的類加載器實現AOP只能攔截本身加載的字節碼,那麼有沒有一種方式可以監控全部類加載器加載字節碼呢?有,使用Instrumentation,它是 Java 5 提供的新特性,使用 Instrumentation,開發者能夠構建一個字節碼轉換器,在字節碼加載前進行轉換。本節使用Instrumentation和javassist來實現AOP。
3.4.1 構建字節碼轉換器
首先須要建立字節碼轉換器,該轉換器負責攔截Business類,並在Business類的doSomeThing方法前使用javassist加入記錄日誌的代碼。
- public class MyClassFileTransformer implements ClassFileTransformer {
-
- /**
- * 字節碼加載到虛擬機前會進入這個方法
- */
- @Override
- public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
- ProtectionDomain protectionDomain, byte[] classfileBuffer)
- throws IllegalClassFormatException {
- System.out.println(className);
- //若是加載Business類才攔截
- if (!"model/Business".equals(className)) {
- return null;
- }
-
- //javassist的包名是用點分割的,須要轉換下
- if (className.indexOf("/") != -1) {
- className = className.replaceAll("/", ".");
- }
- try {
- //經過包名獲取類文件
- CtClass cc = ClassPool.getDefault().get(className);
- //得到指定方法名的方法
- CtMethod m = cc.getDeclaredMethod("doSomeThing");
- //在方法執行前插入代碼
- m.insertBefore("{ System.out.println(\"記錄日誌\"); }");
- return cc.toBytecode();
- } catch (NotFoundException e) {
- } catch (CannotCompileException e) {
- } catch (IOException e) {
- //忽略異常處理
- }
- return null;
- }
3.4.2 註冊轉換器
使用premain函數註冊字節碼轉換器,該方法在main函數以前執行。
- public class MyClassFileTransformer implements ClassFileTransformer {
- public static void premain(String options, Instrumentation ins) {
- //註冊我本身的字節碼轉換器
- ins.addTransformer(new MyClassFileTransformer());
- }
- }
3.4.3 配置和執行
須要告訴JVM在啓動main函數以前,須要先執行premain函數。首先須要將premain函數所在的類打成jar包。並修改該jar包裏的META-INF\MANIFEST.MF 文件。
- Manifest-Version: 1.0
- Premain-Class: bci. MyClassFileTransformer
而後在JVM的啓動參數里加上。-javaagent:D:\java\projects\opencometProject\Aop\lib\aop.jar
3.4.4 輸出
執行main函數,你會發現切入的代碼無侵入性的織入進去了。
- public static void main(String[] args) {
- new Business().doSomeThing();
- new Business().doSomeThing2();
- }
-
輸出
- model/Business
- sun/misc/Cleaner
- java/lang/Enum
- model/IBusiness
- model/IBusiness2
- 記錄日誌
- 執行業務邏輯
- 執行業務邏輯2
- java/lang/Shutdown
- java/lang/Shutdown$Lock
從輸出中能夠看到系統類加載器加載的類也通過了這裏。
4 AOP實戰
說了這麼多理論,那AOP到底能作什麼呢? AOP能作的事情很是多。
- 性能監控,在方法調用先後記錄調用時間,方法執行太長或超時報警。
- 緩存代理,緩存某方法的返回值,下次執行該方法時,直接從緩存裏獲取。
- 軟件破解,使用AOP修改軟件的驗證類的判斷邏輯。
- 記錄日誌,在方法執行先後記錄系統日誌。
- 工做流系統,工做流系統須要將業務代碼和流程引擎代碼混合在一塊兒執行,那麼咱們可使用AOP將其分離,並動態掛接業務。
- 權限驗證,方法執行前驗證是否有權限執行當前方法,沒有則拋出沒有權限執行異常,由業務代碼捕捉。
4.1 Spring的AOP
Spring默認採起的動態代理機制實現AOP,當動態代理不可用時(代理類無接口)會使用CGlib機制。但Spring的AOP有必定的缺點,第一個只能對方法進行切入,不能對接口,字段,靜態代碼塊進行切入(切入接口的某個方法,則該接口下全部實現類的該方法將被切入)。第二個同類中的互相調用方法將不會使用代理類。由於要使用代理類必須從Spring容器中獲取Bean。第三個性能不是最好的,從3.3章節咱們得知使用自定義類加載器,性能要優於動態代理和CGlib。
能夠獲取代理類
- public IMsgFilterService getThis()
- {
- return (IMsgFilterService) AopContext.currentProxy();
- }
-
- public boolean evaluateMsg () {
- // 執行此方法將織入切入邏輯
- return getThis().evaluateMsg(String message);
- }
-
- @MethodInvokeTimesMonitor("KEY_FILTER_NUM")
- public boolean evaluateMsg(String message) {
不能獲取代理類
Java代碼
- public boolean evaluateMsg () {
-
- return evaluateMsg(String message);
- }
-
- @MethodInvokeTimesMonitor("KEY_FILTER_NUM")
- public boolean evaluateMsg(String message) {