解釋AOP
- AOP(Aspect-OrientedProgramming,面向方面編程),可以說是OOP(Object-Oriented Programing,面向對象編程)的補充和完善。OOP允許你定義從上到下的關系,但并不適合定義從左到右的關系。
- 將那些與業務無關,卻為業務模塊所共同調用的邏輯或責任封裝起來,便于減少系統的重復代碼,降低模塊間的耦合度,并有利于未來的可操作性和可維護性。
傳統編碼方式與使用aop的區別
兩種代理:
- JDK:基于接口
- 原理:動態代理 + 反射
- 定義一個類XxHandler,實現InvocationHandler接口,里邊包含真實對象的實例(實際上是一個Object,具體真實對象在運行期賦值給Object),并使用反射調用真實對象的指定方法
- 使用Proxy.newProxyInstance(ClassLoader, interfaces, InvocationHandler)創建代理對象,之后調用相應的真實對象的方法即可。由于該方法的第二個對象是interface,所以JDK只能基于接口實現動態代理。
- CGLib:基于類
- 原理:動態生成字節碼技術,即為將要攔截的類動態生成子類,然后在子類中攔截所有父類的調用并順勢織入橫切邏輯。
- 由于是創建子類,所以不能代理目標類中的private和final方法。
比較:
- CGLib創建代理對象花費的時間長于JDK,因為要動態生成子類
- CGLib創建好的代理對象的性能由于JDK
術語
切點(Pointcut)、連接點(Join point)、切面(Aspect)、引入(Introduction)、織入(Weaving)、通知(Advice)。
Aspect 切面
AOP核心就是切面,它將多個類的通用行為封裝成可重用的模塊,該模塊含有一組API提供橫切功能。比如,一個日志模塊可以被稱作日志的AOP切面。根據需求的不同,一個應用程序可以有若干切面。在Spring AOP中,切面通過帶有@Aspect注解的類實現。
在Spring AOP 中,關注點和橫切關注的區別是什么
- 關注點是應用中一個模塊的行為,一個關注點可能會被定義成一個我們想實現的一個功能。
- 橫切關注點是一個關注點,此關注點是整個應用都會使用的功能,并影響整個應用,比如日志,安全和數據傳輸,幾乎應用的每個模塊都需要的功能。因此這些都屬于橫切關注點。
連接點
連接點代表一個應用程序的某個位置,在這個位置我們可以插入一個AOP切面,它實際上是個應用程序執行Spring AOP的位置。
通知
- 通知是個在方法執行前或執行后要做的動作,實際上是程序執行時要通過SpringAOP框架觸發的代碼段。
- Spring切面可以應用五種類型的通知:
- before:前置通知,在一個方法執行前被調用。
- after: 在方法執行之后調用的通知,無論方法執行是否成功。
- after-returning: 僅當方法成功完成后執行的通知。
- after-throwing: 在方法拋出異常退出時執行的通知。
- around: 在方法執行之前和之后調用的通知。
切點
切入點是一個或一組連接點,通知將在這些位置執行??梢酝ㄟ^表達式或匹配的方式指明切入點。
什么是引入?
引入允許我們在已存在的類中增加新的方法和屬性。
什么是目標對象?
被一個或者多個切面所通知的對象。它通常是一個代理對象。也指被通知(advised)對象。
什么是代理?
代理是通知目標對象后創建的對象。從客戶端的角度看,代理對象和目標對象是一樣的。
有幾種不同類型的自動代理?
- BeanNameAutoProxyCreator
- DefaultAdvisorAutoProxyCreator
- Metadata autoproxying
什么是織入
織入是將切面和到其他應用類型或對象連接或創建一個被通知對象的過程。
織入可以在編譯時,加載時,或運行時完成。
切面例子
@Aspect
public class TransactionDemo {
@Pointcut(value="execution(* com.yangxin.core.service.*.*.*(..))")
public void point(){
}
@Before(value="point()")
public void before(){
System.out.println("transaction begin");
}
@AfterReturning(value = "point()")
public void after(){
System.out.println("transaction commit");
}
@Around("point()")
public void around(ProceedingJoinPoint joinPoint) throws Throwable{
System.out.println("transaction begin");
joinPoint.proceed();
System.out.println("transaction commit");
}
}
spring aop原理
AOP 代理其實是由 AOP 框架動態生成的一個對象,該對象可作為目標對象使用。AOP 代理包含了目標對象的全部方法,但 AOP 代理中的方法與目標對象的方法存在差異:AOP 方法在特定切入點添加了增強處理,并回調了目標對象的方法。
Spring 的 AOP 代理由 Spring 的 IoC 容器負責生成、管理,其依賴關系也由 IoC 容器負責管理。因此,AOP 代理可以直接使用容器中的其他 Bean 實例作為目標,這種關系可由 IoC 容器的依賴注入提供。
aop開發時,其中需要程序員參與的只有 3 個部分:
- 定義普通業務組件。
- 定義切入點,一個切入點可能橫切多個業務組件。
- 定義增強處理,增強處理就是在 AOP 框架為普通業務組件織入的處理動作。
Spring提供了兩種方式來生成代理對象: JDKProxy和Cglib,具體使用哪種方式生成由AopProxyFactory根據AdvisedSupport對象的配置來決定。默認的策略是如果目標類是接口,則使用JDK動態代理技術,否則使用Cglib來生成代理。
JDK動態代理
JDK動態代理主要涉及到java.lang.reflect包中的兩個類:Proxy和InvocationHandler。InvocationHandler是一個接口,通過實現該接口定義橫切邏輯,并通過反射機制調用目標類的代碼,動態將橫切邏輯和業務邏輯編制在一起。
Proxy利用InvocationHandler動態創建一個符合某一接口的實例,生成目標類的代理對象。CGLib動態代理
CGLib全稱為Code Generation Library,是一個強大的高性能,高質量的代碼生成類庫,可以在運行期擴展Java類與實現Java接口,CGLib封裝了asm,可以再運行期動態生成新的class。和JDK動態代理相比較:JDK創建代理有一個限制,就是只能為接口創建代理實例,而對于沒有通過接口定義業務方法的類,則可以通過CGLib創建動態代理。
aop與攔截器、過濾器是什么關系
先來回顧一下攔截器與過濾器。如下圖一網友的測試,在web.xml中注冊了TestFilter1和TestFilter2。然后在spring的配置文件中配置了BaseInterceptor和TestInterceptor。得到的結果如下圖所示。從圖中可以看出,攔截器和過濾器都橫切了業務方法,看似符合aop的思想。
- Filter過濾器:攔截web訪問url地址
- Interceptor攔截器:攔截以 .action結尾的url,攔截Action的訪問
- Spring AOP攔截器:只能攔截Spring管理Bean的訪問(業務層Service)