如何使用Spring AOP的通知類型及創(chuàng)建通知
寫在最前端
1.SpringAOP中共有六種通知類型,只要我們自定義一個類實(shí)現(xiàn)對應(yīng)的接口,它們?nèi)际莖rg.springframework.aop包中的。
2.AOP的連接點(diǎn)可以是方法調(diào)用、方法調(diào)用本身、類初始化、對象實(shí)例化時,但是SpringAOP中全是方法調(diào)用,更簡單,也最實(shí)用
| 通知名稱 | 接口 |
|---|---|
| 前置通知 | org.springframework.aop.MethodBeforeAdvice |
| 后置返回通知 | org.springframework.aop.AfterReturningAdvice |
| 后置通知 | org.springframework.aop.AfterAdvice |
| 環(huán)繞通知 | org.springframework.aop.MethodInterceptor |
| 異常通知 | org.springframework.aop.ThrowsAdvice |
| 引入通知 | org.springframework.aop.IntroductionInterceptor |
寫一個公共類,用于目標(biāo)對象
public class Person {
private String name;
public boolean saySomething(String something){
System.out.println("Pereson類中說了一句:"+something);
return true;//默認(rèn)返回true
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
一、創(chuàng)建前置通知(也就是目標(biāo)方法調(diào)用前執(zhí)行)
前置通知可以修改傳遞給方法的參數(shù),并且可以通過拋出異常來阻止方法的執(zhí)行,可以用前置通知實(shí)現(xiàn)用戶登錄的驗(yàn)證,SpringSecurity就是這么做的
1.例子:在一個方法執(zhí)行前將包含方法名稱的消息寫入到控制臺中,并且將傳入的參數(shù)修改下。(文章中寫的內(nèi)容比較小,大多數(shù)在代碼中有注釋,大家可以下載代碼查看)
/**
* 前置通知類
*/
public class BeforeAdvice implements MethodBeforeAdvice {
@Override
public void before(Method method, Object[] objects, @Nullable Object o) throws Throwable {
//第一個參數(shù)是目標(biāo)方法對象,第二個是參數(shù),第三個是做為調(diào)用目標(biāo)的object(這是personr實(shí)例)
//打印方法名
System.out.println("要執(zhí)行的方法是:"+method.getName());
//修改參數(shù)為lyn4ever
objects[0]="lyn4ever";//我們修改成為了lyn4ever,所以打印出來的就是lyn4ever,而不是zhangsan
}
public static void main(String[] args) {
Person person = new Person();
ProxyFactory pf =new ProxyFactory();
pf.addAdvice(new BeforeAdvice());
pf.setTarget(person);
Person proxy = (Person) pf.getProxy();
//我這里傳的參數(shù)是zhangsan,理論上它應(yīng)該打印出來zhangsan
proxy.saySomething("zhangsan");
}
}

沒毛病,本來我輸入的是zhangsan,在aop中將參數(shù)改為了lyn4ever,這樣就完美的替換了。
二、后置返回通知
是在連接點(diǎn)(方法調(diào)用)返回后執(zhí)行,這顯然不能像上邊那樣修改參數(shù),也不能修改返回值。但是可以拋出可以發(fā)送到堆棧的異常,同樣也可以調(diào)用其他方法。
/**
* 后置返回通知
*/
public class AfterReturnAdvice implements AfterReturningAdvice {
@Override
public void afterReturning(@Nullable Object o, Method method, Object[] objects, @Nullable Object o1) throws Throwable {
/*
參數(shù)和前置通知是一樣的
這個是在返回之后調(diào)用,因此,person中的saySomething會先打印,我們在這里修改的參數(shù)不起作任何作用
*/
System.out.println("調(diào)用的方法是:"+method.getName()+"這句是在saySomething之后");//這句是在saySomething之后
objects[0]="lyn4ever";//這句可以修參數(shù),但是之前的方法已經(jīng)執(zhí)行過了,所以不起任何作用
System.out.println("我們修改了參數(shù)為:"+objects[0]+"但是沒有任何用");//這時候這個參數(shù)并不會傳到person.saysomething(),因?yàn)橐呀?jīng)調(diào)用過了
}
public static void main(String[] args) {
Person person = new Person();
ProxyFactory pf = new ProxyFactory();
pf.addAdvice(new AfterReturnAdvice());//注意修改這個為當(dāng)前類中的通知類
pf.setTarget(person);
Person proxy = (Person) pf.getProxy();
proxy.saySomething("zhangsan");
}
}

三、環(huán)繞通知
這人最好理解了,就是在方法調(diào)用前后都可以執(zhí)行代碼??雌饋硐袷乔爸煤蠛笾玫募希撬梢孕薷姆椒ǖ姆祷刂担?yàn)樗鼘?shí)現(xiàn)的invoke方法的返回值是Object,所以我們就可以修改,而前置通知的返回是void,所以沒法修改的。甚至以至于我們可以不調(diào)用目標(biāo)對象中的連接點(diǎn)方法,我們完全修改這個方法的全部代碼。
public class MyMethodInterceptor implements MethodInterceptor {
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
return null;
}
}
雖然這個invoke()方法中并沒有提供像之前的那些參數(shù),但是這一個invocation實(shí)例可以得到


代碼示例
/**
* 環(huán)繞通知
*/
public class MyMethodInterceptor implements MethodInterceptor {
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
//在這個invoation中有一切我們想要的方法相關(guān)
System.out.println("類名是:"+invocation.getThis().getClass().getName());
System.out.println("目標(biāo)方法是:"+invocation.getMethod().getName());
Object[] arguments = invocation.getArguments();//這個就是參數(shù)
System.out.println("第一個參數(shù) 是:"+arguments[0]);
//我們修改第一個參數(shù)為lyn4ever
arguments[0]="lyn4ever";
invocation.proceed();//執(zhí)行目標(biāo)方法
System.out.println("這個是在之后執(zhí)行的");
return false;//修改返回值
}
public static void main(String[] args) {
Person person = new Person();
ProxyFactory pf = new ProxyFactory();
pf.addAdvice(new MyMethodInterceptor());//注意修改這個為當(dāng)前類中的通知類
pf.setTarget(person);
Person proxy = (Person) pf.getProxy();
boolean flag = proxy.saySomething("zhangsan");
System.out.println(flag);//方法本來是要返回true的
}
}
可以看到,我們修改了目標(biāo)方法返回的值。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。
相關(guān)文章
MyBatisPlus的IService接口實(shí)現(xiàn)
MyBatisPlus是一個為MyBatis提供增強(qiáng)的工具,它通過IService接口簡化了數(shù)據(jù)庫的CRUD操作,IService接口封裝了一系列常用的數(shù)據(jù)操作方法,本文就來介紹一下,感興趣的可以了解一下2024-10-10
springboot prototype設(shè)置多例不起作用的解決操作
這篇文章主要介紹了springboot prototype設(shè)置多例不起作用的解決操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09
Spring中的@AutoWired與@Resource及@Qualifier注解詳解
這篇文章主要介紹了Spring中的@AutoWired與@Resource及@Qualifier注解詳解,spring不但支持自己定義的@Autowired注解,所以Autowired與Spring是強(qiáng)相關(guān)性,只能在spring框架中使用,而后幾個注解則不然,需要的朋友可以參考下2023-11-11
聊聊Spring循環(huán)依賴三級緩存是否可以減少為二級緩存的情況
這篇文章主要介紹了聊聊Spring循環(huán)依賴三級緩存是否可以減少為二級緩存的情況,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2021-02-02
java實(shí)現(xiàn)上傳網(wǎng)絡(luò)圖片到微信臨時素材
這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)上傳網(wǎng)絡(luò)圖片到微信臨時素材,網(wǎng)絡(luò)圖片上傳到微信服務(wù)器,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-07-07
java發(fā)送kafka事務(wù)消息的實(shí)現(xiàn)方法
本文主要介紹了java發(fā)送kafka事務(wù)消息的實(shí)現(xiàn)方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-07-07

