目錄
1、介紹AOP
1.1、AOP的定義
1.2、AOP的作用
1.3、AOP的核心概念及術語
2、AOP實現示例
3、@EnableAspectJAutoProxy注解
1、介紹AOP
1.1、AOP的定義
????????AOP(Aspect Orient Programming),直譯過來就是面向切面編程,AOP 是一種編程思想,是面向對象編程(OOP)的一種補充。AOP可以攔截指定的方法并且對方法增強,而且無需侵入到業務代碼中,使業務與非業務處理邏輯分離,比如Spring的事務,通過事務的注解配置,Spring會自動在業務方法中開啟、提交業務,并且在業務處理失敗時,執行相應的回滾策略。
1.2、AOP的作用
作用:AOP 采取橫向抽取機制(動態代理),取代了傳統縱向繼承機制的重復性代碼,其應用主要體現在事務處理、日志管理、權限控制、異常處理等方面。
主要作用是分離功能性需求和非功能性需求,使開發人員可以集中處理某一個關注點或者橫切邏輯,減少對業務代碼的侵入,增強代碼的可讀性和可維護性。
簡單的說,AOP 的作用就是保證開發者在不修改源代碼的前提下,為系統中的業務組件添加某種通用功能。
AOP的經典應用場景有:日志記錄、事務管理、權限驗證、性能監測
AOP可以攔截指定的方法,并且對方法增強,比如:事務、日志、權限、性能監測等增強,而且無需侵入到業務代碼中,使業務與非業務處理邏輯分離。
1.3、AOP的核心概念及術語
目標對象(Target):目標對象指將要被增強(比如增強功能)的對象,或者說被一個或多個切面所通知的對象,即包含主業務邏輯的類的對象,即為目標對象。
可以說要增強的對象(通常會有很多個)就是目標對象
切面(Aspect):指關注點模塊化,這個關注點可能會橫切多個對象。事務管理是企業級Java應用中有關橫切關注點的例子。 在Spring AOP中,切面可以使用通用類基于模式的方式(schema-basedapproach)或者在普通類中以@Aspect注解(@Aspect注解方式)來實現。
可以說增強代碼放入的那個類就叫切面類,由@Aspect注解來標識
通知(Advice):在切面的某個特定的連接點上執行的動作。通知有多種類型,包括“around”,“before” 和“after”等等。通知的類型將在后面詳細了解。許多AOP框架,包括Spring在內,都是以攔截器做通知模型的,并維護著一個以連接點為中心的攔截器鏈。
用來放增強的代碼的那個方法就是通知
- 環繞通知@Around:可以把代碼增強在目標方法的任意地方,更通用
- 前置通知@Before:目標方法之前執行
- 后置通知@After:目標方法之后執行
- 異常通知@AfterThrowing:目標方法出現了異常執行
- 返回通知@AfterReturning:目標方法返回值執行
切點(Pointcut):匹配連接點的斷言。通知和切點表達式相關聯,并在滿足這個切點的連接點上運行(例如,當執行某個特定名稱的方法時)。切點表達式如何和連接點匹配是AOP的核心:Spring默認使用Aspectj切點語義。
增強代碼要對哪些類中的哪些方法進行增強,進行切割,指的是被增強的方法,即要切哪些東西。切點表達式
連接點(Join point):在Spring AOP中,一個連接點總是代表一個方法的執行,其實就代表增強的方法。
連接點就是通知和目標方法的一個橋梁,可以獲取到目標方法的信息,就得通過JoinPoint
顧問(Advisor):顧問是Advice的一種包裝體現,Advisor是Pointcut以及Advice的一個結合,用來管理Advice和Pointcut。應用無需關心.
顧問是源碼中的體現,會封裝切點和通知
織入(Weaving):將通知切入連接點的過程叫織入,就是我這個增強代碼織入到目標方法或者叫切入到目標方法都可以
2、AOP實現示例
????????假設有一個類的所有業務功能都已經實現,但是最后我想在所有的功能里面加一個公共的功能(記錄所有的請求的用時), 這時候就可以利用AOP來實現,切面就是(需要增強的公共的、跟業務沒有關系的公共代碼)。可以在不改變原有業務代碼的基礎上進行了增強。所以面向切面編程往往讓我們的開發更加低耦合,也大大減少了代碼量,同時呢讓我們更專注于業務模塊的開發,把那些與業務無關的東西提取出去,便于后期的維護和迭代。
如何在Spring中創建一個所謂切面?
切面里面的代碼怎么運行在業務方法(運行在之前、運行在之后)
使用AOP需要引入依賴
<!--以spring-boot-starter為前綴的依賴都是springboot官方提供的依賴,
不需要自己來指定版本,因為parent都幫我們管理好了,
springboot通過這種方式統一管理了版本,可以減少版本的沖突-->
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId>
</dependency>
aspectj:AOP概念并不是spring獨家提出來的,AOP早就有成熟框架已經實現了,aspectj就是一個成熟的AOP的解決方案,spring只不過是把aspectj集成進來了
AOP實現示例代碼:@Aspect注解可以把一個類標記為切面類
自定義類UserService,假設其所有業務功能(增刪查改)都已經實現,但是我想給這些功能加一個記錄請求用時的功能。這時用AOP來實現
//自定義類UserService,假設其所有業務功能都已經實現
@Service
public class UserService {// 增加增刪改查方法,這里就用一個打印語句來代替具體的方法功能了public void add(){System.out.println("增加");}public void delete(){System.out.println("刪除");}public void update(){System.out.println("修改");}public void query(){System.out.println("查詢");}
}//切面類LogAspect,通過這個切面類來實現在所有的功能里面加一個公共的功能(記錄所有的請求的用時)
@Aspect //把這個類標記為切面類
@Component //切面類必須聲明為Spring的bean
public class LogAspect, {// execution(* com.lt.UserService.*(..)) 切點表達式// @Around 環繞通知@Around("execution(* com.lt.UserService.*(..))") //切點表達式public void log(ProceedingJoinPoint proceedingJoinPoint){//記錄方法用時long begin = System.currentTimeMillis();// 執行具體的方法try {proceedingJoinPoint.proceed();} catch (Throwable e){System.out.println("方法執行異常:" + e.getMessage());}long end = System.currentTimeMillis();System.out.println("方法用時:" + (end - begin) + "ms");}}//測試類C4AopApplicationTests
@SpringBootTest(classes = C4AopApplicationTests.class)
@ComponentScan
class C4AopApplicationTests {@Testvoid contextLoads(@Autowired UserService userService) {userService.add();}}
運行結果:
3、@EnableAspectJAutoProxy注解
1、@EnableAspectJAutoProxy:啟用AOP,沒有這個注解AOP功能無法使用
如果要使用SpringAOP的功能,必須要添加一個@EnableAspectJAutoProxy注解,有了這個注解才能支持@Aspect等相關的一系列AOP注解的功能,這個注解就相當于在傳統的xml配置文件中添加 <aop:aspectj-autoproxy>一樣。
2、@EnableAspectAutoProxy不是一定要加的
比如上面的AOP示例,為什么沒有加@EnableAspectJAutoProxy注解,AOP也起作用了?
這是因為SpringBoot其實會通過啟動類自動幫我們加上@EnableAspectJAutoProxy,所以可以省略。(@SpringBootApplication這個注解幫我們做了很多的事情,包括自動幫我們加上@EnableAspectJAutoProxy注解)
//測試類C4AopApplicationTests
@SpringBootTest(classes = C4AopApplicationTests.class)
@ComponentScan
class C4AopApplicationTests {......
}//啟動類
@SpringBootApplication // 由于這個注解起了作用,所以自動加了@EnableAspectJAutoProxy
public class C4AopApplication {......
}
所以com.lt包下的啟動類C4AopApplication和他上面的上面的注解@SpringBootApplicatio也會被掃描到,由于這個注解起了作用,所以自動加了@EnableAspectJAutoProxy
但是依然建議加上@EnableAspectJAutoProxy,因為在Spring中沒有這個注解AOP功能會無法使用。
?推薦:
【Spring】IOC/DI中常用的注解@Lazy、@Scope與@Conditional-CSDN博客https://blog.csdn.net/m0_65277261/article/details/138277932?spm=1001.2014.3001.5501
【Spring】IOC/DI中常用的注解@Order與@DependsOn-CSDN博客https://blog.csdn.net/m0_65277261/article/details/138167160?spm=1001.2014.3001.5501
【spring】Bean的生命周期回調函數和Bean的循環依賴-CSDN博客https://blog.csdn.net/m0_65277261/article/details/138503989?spm=1001.2014.3001.5501