一、什么是AOP?
Aspect Oriented Programing 面向切面編程,相比較 oop 面向?qū)ο缶幊虂碚f,Aop關(guān)注的不再是程
序代碼中某個(gè)類,某些方法,而aop考慮的更多的是一種面到面的切入,即層與層之間的一種切入,所
以稱之為切面。聯(lián)想大家吃的漢堡(中間夾肉)。那么aop是怎么做到攔截整個(gè)面的功能呢?考慮前面
學(xué)到的servlet filter /* 的配置 ,實(shí)際上也是aop 的實(shí)現(xiàn)。
二、AOP能做什么?
AOP主要應(yīng)用于日志記錄,性能統(tǒng)計(jì),安全控制,事務(wù)處理等方面,實(shí)現(xiàn)公共功能性的重復(fù)使用。
三、AOP的特點(diǎn)
1. 降低模塊與模塊之間的耦合度,提高業(yè)務(wù)代碼的聚合度。(高內(nèi)聚低耦合)
2. 提高了代碼的復(fù)用性。
3. 提高系統(tǒng)的擴(kuò)展性。(高版本兼容低版本)
4. 可以在不影響原有的功能基礎(chǔ)上添加新的功能
四、AOP的理解和關(guān)鍵詞
1. 面向切面,相比oop 關(guān)注的是代碼中的層 或面
2. 解耦,提高系統(tǒng)擴(kuò)展性
3. 提高代碼復(fù)用
Aop 關(guān)鍵詞
1. 連接點(diǎn):每一個(gè)方法
2. 切入點(diǎn):匹配的方法集合
3. 切面:連接點(diǎn)與切入點(diǎn)的集合決定了切面,橫切關(guān)注點(diǎn)的抽象
4. 通知:幾種通知
5. 目標(biāo)對(duì)象:被代理對(duì)象
6. 織入:程序運(yùn)行期將切面應(yīng)用到目標(biāo)對(duì)象 并生成代理對(duì)象的過程
7. 引入:在不修改原始代碼情況下,在程序運(yùn)行期為程序動(dòng)態(tài)引入方法或字段的過程
五、Spring AOP的實(shí)現(xiàn)
5.1 Spring AOP環(huán)境搭建
坐標(biāo)依賴引入
<!--Spring AOP-->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.8.9</version>
</dependency>
添加spring.xml的配置
添加命名空間
xmlns:aop="http://www.springframework.org/schema/aop"
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
5.2 注解實(shí)現(xiàn)
定義切面
/**
* 切面
* 切入點(diǎn)和通知的抽象 (與面向?qū)ο笾械?類 相似)
* 定義 切入點(diǎn)和通知 (切入點(diǎn)定義了要攔截哪些類的哪些方法,通知?jiǎng)t定義了攔截過方法后要做什么)
*/
@Component // 將對(duì)象交給IOC容器去實(shí)例化
@Aspect // 聲明當(dāng)前類是一個(gè)切面
public class LogCut {
/**
* 切入點(diǎn):
* 匹配規(guī)則。規(guī)定什么方法被攔截、需要處理什么方法
* 定義切入點(diǎn)
* @Pointcut("匹配規(guī)則")
* *
Aop 切入點(diǎn)表達(dá)式簡介
* 1. 執(zhí)行任意公共方法:
* execution(public *(..))
* 2. 執(zhí)行任意的set方法
* execution(* set*(..))
* 3. 執(zhí)行com.xxxx.service包下任意類的任意方法
* execution(* com.xxxx.service.*.*(..))
* 4. 執(zhí)行com.xxxx.service 包 以及子包下任意類的任意方法
* execution(* com.xxxx.service..*.*(..))
* *
注:表達(dá)式中的第一個(gè)* 代表的是方法的修飾范圍
* 可選值:private、protected、public (* 表示所有范圍)
*/
@Pointcut("execution (* com.xxxx.service..*.*(..) )")
public void cut(){}
/**
* 聲明前置通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法執(zhí)行前 執(zhí)行該通知
* *
/
@Before(value = "cut()")
public void before() {
System.out.println("前置通知.....");
} /
**
* 聲明返回通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法(無異常)執(zhí)行后 執(zhí)行該通知
* *
/
@AfterReturning(value = "cut()")
public void afterReturn() {
System.out.println("返回通知.....");
} /
**
* 聲明最終通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法(無異常或有異常)執(zhí)行后 執(zhí)行該通知
* *
/
@After(value = "cut()")
public void after() {
System.out.println("最終通知.....");
} /
**
* 聲明異常通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法出現(xiàn)異常時(shí) 執(zhí)行該通知
*/
@AfterThrowing(value="cut()",throwing = "e")
public void afterThrow(Exception e) {
System.out.println("異常通知....." + " 異常原因:" + e.getCause());
} /
**
* 聲明環(huán)繞通知 并將通知應(yīng)用到切入點(diǎn)上
* 方法執(zhí)行前后 通過環(huán)繞通知定義相應(yīng)處理
* 需要通
* @param pjp
* @return
*/
@Around(value = "cut()")
public Object around(ProceedingJoinPoint pjp) {
System.out.println("前置通知...");
Object object = null;
try {
object = pjp.proceed();
System.out.println(pjp.getTarget() + "======" + pjp.getSignature());
// System.out.println("返回通知...");
} catch (Throwable throwable) {
throwable.printStackTrace();
System.out.println("異常通知...");6.2.2. 配置文件(spring.xml)
6.3. XML實(shí)現(xiàn)
6.3.1. 定義切面
}
S
ystem.out.println("最終通知...");
return object;
}
}
配置文件(spring.xml)
<!--配置AOP代理-->
<aop:aspectj-autoproxy/>
5.3 XML實(shí)現(xiàn)
定義切面
**
* 切面
* 切入點(diǎn)和通知的抽象 (與面向?qū)ο笾械?類 相似)
* 定義 切入點(diǎn)和通知 (切入點(diǎn)定義了要攔截哪些類的哪些方法,通知?jiǎng)t定義了攔截過方法后要做什么)
*/
@Component // 將對(duì)象交給IOC容器去實(shí)例化
public class LogCut02 {
public void cut(){}
/**
* 聲明前置通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法執(zhí)行前 執(zhí)行該通知
* *
/
public void before() {
System.out.println("前置通知.....");
} /
**
* 聲明返回通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法(無異常)執(zhí)行后 執(zhí)行該通知
* *
/
public void afterReturn() {
System.out.println("返回通知.....");
} /
**
* 聲明最終通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法(無異常或有異常)執(zhí)行后 執(zhí)行該通知
* *
/6.3.2. 配置文件(spring.xml)
public void after() {
System.out.println("最終通知.....");
} /
**
* 聲明異常通知 并將通知應(yīng)用到定義的切入點(diǎn)上
* 目標(biāo)類方法出現(xiàn)異常時(shí) 執(zhí)行該通知
*/
public void afterThrow(Exception e) {
System.out.println("異常通知....." + " 異常原因:" + e.getCause());
} /
**
* 聲明環(huán)繞通知 并將通知應(yīng)用到切入點(diǎn)上
* 方法執(zhí)行前后 通過環(huán)繞通知定義相應(yīng)處理
* 需要通過顯式調(diào)用對(duì)應(yīng)的方法,否則無法訪問指定方法 (pjp.proceed();)
* @param pjp
* @return
*/
public Object around(ProceedingJoinPoint pjp) {
System.out.println("前置通知...");
Object object = null;
try {
object = pjp.proceed();
System.out.println(pjp.getTarget() + "======" + pjp.getSignature());
// System.out.println("返回通知...");
} catch (Throwable throwable) {
throwable.printStackTrace();
System.out.println("異常通知...");
} S
ystem.out.println("最終通知...");
return object;
}
}
配置文件(spring.xml)
<!--aop相關(guān)配置-->
<aop:config>
<!--aop切面-->
<aop:aspect ref="logCut02">
<!-- 定義aop 切入點(diǎn) -->
<aop:pointcut id="cut" expression="execution(* com.xxxx.service..*.*
(..))"/>
<!-- 配置前置通知 指定前置通知方法名 并引用切入點(diǎn)定義 -->
<aop:before method="before" pointcut-ref="cut"/>
<!-- 配置返回通知 指定返回通知方法名 并引用切入點(diǎn)定義 -->
<aop:after-returning method="afterReturn" pointcut-ref="cut"/>
<!-- 配置異常通知 指定異常通知方法名 并引用切入點(diǎn)定義 -->7. Spring AOP總結(jié)
7.1. 代理模式實(shí)現(xiàn)三要素
1. 接口定義
2. 目標(biāo)對(duì)象與代理對(duì)象必須實(shí)現(xiàn)統(tǒng)一接口
3. 代理對(duì)象持有目標(biāo)對(duì)象的引用 增強(qiáng)目標(biāo)對(duì)象行為
7.2. 代理模式實(shí)現(xiàn)分類以及對(duì)應(yīng)區(qū)別
1. 靜態(tài)代理:手動(dòng)為目標(biāo)對(duì)象制作代理對(duì)象,即在程序編譯階段完成代理對(duì)象的創(chuàng)建
2. 動(dòng)態(tài)代理:在程序運(yùn)行期動(dòng)態(tài)創(chuàng)建目標(biāo)對(duì)象對(duì)應(yīng)代理對(duì)象。
3. jdk動(dòng)態(tài)代理:被代理目標(biāo)對(duì)象必須實(shí)現(xiàn)某一或某一組接口 實(shí)現(xiàn)方式 通過回調(diào)創(chuàng)建代理對(duì)象。
4. cglib 動(dòng)態(tài)代理:被代理目標(biāo)對(duì)象可以不必實(shí)現(xiàn)接口,繼承的方式實(shí)現(xiàn)。
動(dòng)態(tài)代理相比較靜態(tài)代理,提高開發(fā)效率,可以批量化創(chuàng)建代理,提高代碼復(fù)用率。
7.3. Aop 理解
1. 面向切面,相比oop 關(guān)注的是代碼中的層 或面
2. 解耦,提高系統(tǒng)擴(kuò)展性
3. 提高代碼復(fù)用
7.4. Aop 關(guān)鍵詞
1. 連接點(diǎn):每一個(gè)方法
2. 切入點(diǎn):匹配的方法集合
3. 切面:連接點(diǎn)與切入點(diǎn)的集合決定了切面,橫切關(guān)注點(diǎn)的抽象
4. 通知:幾種通知
5. 目標(biāo)對(duì)象:被代理對(duì)象
6. 織入:程序運(yùn)行期將切面應(yīng)用到目標(biāo)對(duì)象 并生成代理對(duì)象的過程
7. 引入:在不修改原始代碼情況下,在程序運(yùn)行期為程序動(dòng)態(tài)引入方法或字段的過程
<aop:after-throwing method="afterThrow" throwing="e" pointcutref="cut"/>
<!-- 配置最終通知 指定最終通知方法名 并引用切入點(diǎn)定義 -->
<aop:after method="after" pointcut-ref="cut"/>
<!-- 配置環(huán)繞通知 指定環(huán)繞通知方法名 并引用切入點(diǎn)定義 -->
<aop:around method="around" pointcut-ref="cut"/>
</aop:aspect>
</aop:config>
六、Spring AOP總結(jié)
6.1. 代理模式實(shí)現(xiàn)三要素
1. 接口定義
2. 目標(biāo)對(duì)象與代理對(duì)象必須實(shí)現(xiàn)統(tǒng)一接口
3. 代理對(duì)象持有目標(biāo)對(duì)象的引用 增強(qiáng)目標(biāo)對(duì)象行為
6.2. 代理模式實(shí)現(xiàn)分類以及對(duì)應(yīng)區(qū)別
1. 靜態(tài)代理:手動(dòng)為目標(biāo)對(duì)象制作代理對(duì)象,即在程序編譯階段完成代理對(duì)象的創(chuàng)建
2. 動(dòng)態(tài)代理:在程序運(yùn)行期動(dòng)態(tài)創(chuàng)建目標(biāo)對(duì)象對(duì)應(yīng)代理對(duì)象。
3. jdk動(dòng)態(tài)代理:被代理目標(biāo)對(duì)象必須實(shí)現(xiàn)某一或某一組接口 實(shí)現(xiàn)方式 通過回調(diào)創(chuàng)建代理對(duì)象。
4. cglib 動(dòng)態(tài)代理:被代理目標(biāo)對(duì)象可以不必實(shí)現(xiàn)接口,繼承的方式實(shí)現(xiàn)。
動(dòng)態(tài)代理相比較靜態(tài)代理,提高開發(fā)效率,可以批量化創(chuàng)建代理,提高代碼復(fù)用率。






