Java實現AOP代理的三種方式詳解
業務場景:首先你有瞭一個非常好的前輩無時無刻的在“教育”你。有這麼一天,它叫你將它寫好的一個方法進行改進測試,這時出現瞭功能迭代的情況。然後前輩好好“教育”你的說,不行改我的代碼!改就腿打折!悲催的你有兩條路可走,拿出你10年跆拳道的功夫去火拼一波然後拍拍屁股瀟灑走人,要麼就是悲催的開始百度。。。這時你會發現,我擦怎麼把AOP代理這種事給忘瞭?【其實在我們工作中很少去手寫它,但是它又是很常見的在使用(控制臺日志)】
怎麼辦?打贏送手鐲,打輸睡大覺?
兄弟,信我的!
寫吧。。。
AOP是一種設計思想,是軟件設計領域中的面向切面編程,它是面向對象編程的一種補充和完善,它以通過預編譯方式和運行期動態代理方式實現在不修改源代碼的情況下給程序動態統一添加額外功能的一種技術。
利用AOP可以對業務邏輯的各個部分進行隔離,從而使得業務邏輯各部分之間的耦合度降低,提高程序的可重用性,同時提高瞭開發的效率
簡單說就是在不影響原有功能代碼的情況下進行擴展,浸入少。
廢話不多說,開搞!
1、JDK實現
MyAop package com.example.quasar.aop; import java.lang.reflect.InvocationHandler; import java.lang.reflect.Method; //繼承InvocationHandler接口實現 public class MyAop implements InvocationHandler { private Object object; public MyAop(Object object) { this.object = object; } @Override public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { //前置 System.out.println("前置運行"); Object invoke = method.invoke(this.object, args); //後置 System.out.println("後置運行"); return invoke; } }
IndexService
package com.example.quasar.service; public interface IndexService { public void run(); public void run1(); }
IndexServiceImpl
package com.example.quasar.service.impl; import com.example.quasar.service.IndexService; public class IndexServiceImpl implements IndexService { @Override public void run() { System.out.println("運行瞭"); } @Override public void run1() { System.out.println("運行瞭1"); } }
QuasarApplication
package com.example.quasar; import com.example.quasar.aop.MyAop; import com.example.quasar.service.IndexService; import com.example.quasar.service.impl.IndexServiceImpl; import org.springframework.boot.autoconfigure.SpringBootApplication; import java.lang.reflect.Proxy; @SpringBootApplication public class QuasarApplication { public static void main(String[] args) { //實現類實例化 IndexService indexService = new IndexServiceImpl(); //將實例傳入aop MyAop myAop = new MyAop(indexService); //通過Proxy.newProxyInstance實現代理 IndexService o = (IndexService) Proxy.newProxyInstance(IndexServiceImpl.class.getClassLoader(), new Class[]{IndexService.class}, myAop); //執行實現方法 o.run(); o.run1(); } }
執行結果
2、CGLIB實現
如果項目中,已經使用瞭Spring,可以忽略導入上述兩個包。因為在spring-core中已經集成瞭。否則需要引入該包,百度去找。
MyAop
package com.example.quasar.aop; public class MyAop { public void befor() { System.out.println("befor..."); } public void after() { System.out.println("after..."); } }
IndexService
package com.example.quasar.service; public interface IndexService { public void run(); public void run1(); }
IndexServiceImpl
package com.example.quasar.service.impl; import com.example.quasar.service.IndexService; public class IndexServiceImpl implements IndexService { @Override public void run() { System.out.println("運行瞭"); } @Override public void run1() { System.out.println("運行瞭1"); } }
QuasarApplication
package com.example.quasar; import com.example.quasar.aop.MyAop; import com.example.quasar.service.IndexService; import com.example.quasar.service.impl.IndexServiceImpl; import org.springframework.boot.autoconfigure.SpringBootApplication; import org.springframework.cglib.proxy.Enhancer; import org.springframework.cglib.proxy.MethodInterceptor; import org.springframework.cglib.proxy.MethodProxy; import java.lang.reflect.Method; @SpringBootApplication public class QuasarApplication { public static void main(String[] args) { //實現類實例化 IndexService indexService = new IndexServiceImpl(); // 代理類 ,采用cglib,底層創建目標類的子類 MyAop myAop = new MyAop(); // 核心類 Enhancer enhancer = new Enhancer(); //確定父類 enhancer.setSuperclass(indexService.getClass()); enhancer.setCallback(new MethodInterceptor() { @Override public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable { //前置 myAop.after(); // 執行目標類的方法 Object obj = method.invoke(indexService, args); //後置 myAop.befor(); return obj; } }); // 3.4 創建代理 IndexService proxService = (IndexService) enhancer.create(); proxService.run(); proxService.run1(); } }
執行結果
3、boot註解實現【註意隻對bean有效】
MyAop
package com.v1.aop; import org.aspectj.lang.JoinPoint; import org.aspectj.lang.ProceedingJoinPoint; import org.aspectj.lang.annotation.Around; import org.aspectj.lang.annotation.Aspect; import org.aspectj.lang.annotation.Before; import org.springframework.stereotype.Component; @Aspect @Component public class MyAop { //*號作用:com.v1.service.impl下的所有類和所有方法 @Before(value = "within(com.v1.service.impl.*)") public void before(JoinPoint joinPoint) { System.out.println("before開始執行查詢......."); System.out.println("正在執行的目標類是: " + joinPoint.getTarget()); System.out.println("正在執行的目標方法是: " + joinPoint.getSignature().getName()); } //*號作用:com.v1.controller下的所有類和所有方法 @Around(value = "execution(* com.v1.controller.*.*(..))") public Object aroud(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { System.out.println("aroud環繞通知開始......."); System.out.println("執行的目標類 = " + proceedingJoinPoint.getTarget()); System.out.println("執行的目標方法 = " + proceedingJoinPoint.getSignature().getName()); // 必須方法目標方法 Object proceed = proceedingJoinPoint.proceed(); System.out.println("aroud環繞通知結束......."); // 將目標方法的返回值進行返回,否則調用目標方法的方法無法獲取到返回值 return proceed; } }
非常簡單,直接就根據註解進行代理瞭!
註解參數可以具體在咱們csdn平臺去查一下,有很多文章講的很細的!
執行結果
發個請求跑一下
以上就是Java實現AOP代理的三種方式詳解的詳細內容,更多關於Java AOP代理的資料請關註WalkonNet其它相關文章!
推薦閱讀:
- java.lang.OutOfMemoryError: Metaspace異常解決的方法
- Spring中AOP概念與兩種動態代理模式原理詳解
- Spring AOP的底層實現方式-代理模式
- Java 動態代理的多種實現方式
- Java AOP動態代理詳細介紹