假如沒有aop,在做日志處理的時候,我們會在每個方法中添加日志處理,比如
但大多數(shù)的日子處理代碼是相同的,為了實現(xiàn)代碼復(fù)用,我們可能把日志處理抽離成一個新的方法。但是這樣我們?nèi)匀槐仨毷謩硬迦脒@些方法。
但這樣兩個方法就是強(qiáng)耦合的,假如此時我們不需要這個功能了,或者想換成其他功能,那么就必須一個個修改。
通過動態(tài)代理,可以在指定位置執(zhí)行對應(yīng)流程。這樣就可以將一些橫向的功能抽離出來形成一個獨立的模塊,然后在指定位置
插入這些功能。這樣的思想,被稱為面向切面編程,亦即aop。
為了在指定位置執(zhí)行這些橫向的功能,需要知道指定的是什么地方
例如上圖,方法級別的aop實現(xiàn),在一個程序執(zhí)行鏈條中,把method2稱為切點,也就是說在method2執(zhí)行時會執(zhí)行橫切的功能,那么是在method2之前還是之后呢,又是執(zhí)行什么呢?這些都由advice(通知)來指定。advice有5種類型,分別是
通知類型 簡介
before(前置通知) 目標(biāo)方法調(diào)用之前執(zhí)行
after(后置通知) 目標(biāo)方法調(diào)用之后執(zhí)行
after-returning(返回通知) 目標(biāo)方法執(zhí)行成功后執(zhí)行
after-throwing(異常通知) 目標(biāo)方法拋出異常后執(zhí)行
around(環(huán)繞通知) 相當(dāng)于合并了前置和后置
把切點和通知合在一起就是切面了,一個切面指定了在何時何地執(zhí)行何種方法。在spring aop中如此定義這個切面:
1
2
3
4
5
6
7
8
9
10
|
@aspect @component public class useraspect { @before ( "execution(* com.aop.service.impl.userserviceimpl.login(..))" ) public void loginlog(){ system.out.println( "user login" ); } } |
使用注解@aspect將某個特定的類聲明為切面,這樣,該類下的方法就可以聲明為橫向的功能點后插入到指定位置。使用execution表達(dá)式聲明在這個切點,格式如下
第一個位置指定了方法的返回值,*號代表任意類型的返回值,然后是所在的類和方法名,*號同樣代表任意,就是該類中任意的方法,在上一個例子中方法名是login,則是指定了該類中的login方法。然后最后一個參數(shù)是方法入?yún)ⅲ驗閖ava中支持重載,所以這個參數(shù)可以幫助你更精確的進(jìn)行定位。兩點表示任意參數(shù)類型。這樣,execution表達(dá)式告訴了程序該在何地執(zhí)行通知。而被諸如@before注解修飾的方法就是通知的內(nèi)容,也就是做什么。
至此,我們就可以使用spring aop,但是還有兩點需要得到注意
將切面類聲明為一個bean
切點指定的方法所在的類也同樣需由spring注入才能生效
希望本篇文章對您有所幫助
原文鏈接:http://blog.csdn.net/baidu_33403616/article/details/70304051