Spring AOP是Spring的三大基石之一
它究竟解決了什么痛點(diǎn)能獲如此殊榮握侧?
Java在面向?qū)ο蟮氖澜缋餆o限風(fēng)光蚯瞧,oop成就了無數(shù)經(jīng)典的軟件嘿期,它讓我們的軟件更健壯品擎,更易于維護(hù),但是程序猿對軟件的質(zhì)量追求是永無止境的备徐,oop對于業(yè)務(wù)的抽象和封裝完美的無懈可擊然而對于系統(tǒng)層面的一些需求比如系統(tǒng)日志萄传,性能統(tǒng)計(jì)等,分散在軟件的各個角落蜜猾,維護(hù)起來很是不爽秀菱,這種問題的解決確是oop力所不能及的,于是AOP橫空出世先來看一個簡單的小例子蹭睡,體會一下AOP能解決什么問題:如果我們要在A,B,C三個類的do方法中都加入日志功能定義一個日志類Record衍菱,加日志的方法:addRecord
Public class A{
Public void do(){
…
Record.addRecord();//添加日志
}
}
Public class B{
Public void do(){
…
Record.addRecord();//添加日志
}
}
Public class C{
Public void do(){
…
Record.addRecord();//添加日志
}
}
這樣做可以解決問題,但是總感覺有些別扭肩豁,每個類的do方法中都調(diào)用了添加日志的方法脊串,添加日志的方法不是我們的核心業(yè)務(wù),我們卻要去處理它隨著系統(tǒng)越來越完善清钥,類似這樣的非核心業(yè)務(wù)也會越來越多琼锋,比如權(quán)限,異常處理祟昭,性能監(jiān)控等這樣的功能出現(xiàn)在很多類的很多方法中干擾了我們的核心業(yè)務(wù)代碼缕坎,怎么解決呢?AOP就是為此而生:看看AOP是如何解決的篡悟?
從上圖可以看出對于日志記錄谜叹,性能監(jiān)控匾寝,異常處理這樣的非核心功能,單獨(dú)被抽取出來荷腊,與業(yè)務(wù)代碼分離旗吁,橫切在核心業(yè)務(wù)代碼之上
這就是我們通常所說的面向切面編程(AOP),通過一個例子看看他是如何實(shí)現(xiàn)的
創(chuàng)建一個UserDao類:
@Repository
public class UserDao {
public void addUser(){
System.out.println("添加用戶");
}
public void updateUser(){
System.out.println("修改用戶");
}
public void deleteUser(){
System.out.println("刪除用戶");
}
}
創(chuàng)建一個切面類:
@Aspect
public class MyAspectLog {
/**
* 方法執(zhí)行完后執(zhí)行的方法
*/
@After(value="execution(* cn.xh.dao.UserDao.addUser(..))")
public void log(){
System.out.println("記錄日志");
}
}
在spring配置文件中加入:
<!-- 啟動@aspectj的自動代理支持-->
<aop:aspectj-autoproxy />
<!-- 定義aspect類 -->
<bean name="myAspect" class="cn.xh.dao. MyAspectLog "/>
當(dāng)我們創(chuàng)建UserDao的對象userDao調(diào)用addUser方法的時候會打印“添加用戶”,“記錄日志”很神奇吧,究竟發(fā)生了什么停局?明明addUser方法里面只有打印”添加用戶”啊這就是Spring AOP的強(qiáng)大之處很钓,在運(yùn)行時通過動態(tài)代理技術(shù)對UserDao的addUser方法進(jìn)行了增強(qiáng),添加了記錄日志的功能董栽。動態(tài)代理其實(shí)就是在運(yùn)行時動態(tài)的生成目標(biāo)對象的代理對象码倦,在代理對象中對目標(biāo)對象的方法進(jìn)行增強(qiáng),關(guān)于動態(tài)代理技術(shù)我會在另一篇文章中詳細(xì)介紹锭碳,現(xiàn)在先來看一下AOP中幾個重要的概念:
一:通知:就是會在目標(biāo)方法執(zhí)行前后執(zhí)行的方法
上面這個例子中:
@After(value="execution(* cn.xh.dao.UserDao.addUser(..))")
public void log(){
System.out.println("記錄日志");
}
這個方法就是通知袁稽,目標(biāo)方法是UserDao類的addUser(),在addUser執(zhí)行之后執(zhí)行了log方法,所以log方法是后置通知擒抛,通過在方法上加上@After注解來表示推汽。通過通知和目標(biāo)方法的執(zhí)行順序我們可以把通知分為五種:前置通知(before):在目標(biāo)方法執(zhí)行之前執(zhí)行。后置通知(after):在目標(biāo)方法執(zhí)行之后執(zhí)行后置返回通知(after returning):在目標(biāo)方法返回之后執(zhí)行歧沪,先執(zhí)行后置通知再執(zhí)行后置返回通知歹撒。這三種通知的執(zhí)行順序如下:
try{
try{
//@Before
method.invoke(..);
}finally{
//@After
}
//@AfterReturning
}catch(){
//@AfterThrowing
}
異常通知(after throwing):在目標(biāo)方法拋出異常時執(zhí)行環(huán)繞通知(around):在目標(biāo)函數(shù)執(zhí)行中執(zhí)行二:切入點(diǎn):應(yīng)用通知進(jìn)行增強(qiáng)的目標(biāo)方法現(xiàn)在面臨的問題是如何去描述這個需要被增強(qiáng)的目標(biāo)方法,如果只是一個具體的方法需要增強(qiáng)那簡單诊胞,通過類名和方法名找到它就可以了暖夭,但是往往真實(shí)的需求中很多方法需要同樣的通知進(jìn)行增強(qiáng),Spring AOP為我們提供了一個描述方法的語法比如上例中的:
@After(value="execution(* cn.xh.dao.UserDao.addUser(..))")execution(* cn.xh.dao.UserDao.addUser(..)就是用來描述需要應(yīng)用通知的方法的撵孤。這里的含義是cn.xh.dao包UserDao類中的參數(shù)任意迈着,返回值任意的addUser方法。
關(guān)于這種描述更多具體的寫法我會在另一篇文章中詳述邪码。
三:連接點(diǎn):連接點(diǎn)就是可以應(yīng)用通知進(jìn)行增強(qiáng)的方法
因?yàn)镾pring Aop只能針對方法進(jìn)行增強(qiáng)裕菠,所以這里的連接點(diǎn)指的就是方法,一旦連接點(diǎn)被增強(qiáng)闭专,它就成為了切入點(diǎn)奴潘。
如上例中的:
public void addUser(){
System.out.println("添加用戶");
}
public void updateUser(){
System.out.println("修改用戶");
}
public void deleteUser(){
System.out.println("刪除用戶");
}
三個方法都是連接點(diǎn)。
四:切面:是切入點(diǎn)和通知的結(jié)合
可以用切面類來表示:
@Aspect
public class MyAspectLog {
/**
* 方法執(zhí)行完后執(zhí)行的方法
*/
@After(value="execution(* cn.xh.dao.UserDao.addUser(..))")
public void log(){
System.out.println("記錄日志");
}
}
在這個類中既包含了切入點(diǎn)addUser又包含了通知:log().
五:織入:就是通過動態(tài)代理對目標(biāo)對象方法進(jìn)行增強(qiáng)的過程喻圃。
轉(zhuǎn)自:https://www.zhihu.com/question/24863332/answer/863736101