SpringBoot利用切面注解及反射實(shí)現(xiàn)事件監(jiān)聽功能
前言
當(dāng)某個(gè)事件需要被監(jiān)聽的時(shí)候,我們需要去做其他的事前,最簡單的方式就是將自己的業(yè)務(wù) 方法追加到該事件之后。
但是當(dāng)有N多個(gè)這樣的需求的時(shí)候我們都這樣一個(gè)個(gè)去添加修改事件的源碼嗎?
這篇文章將告訴你如何用一個(gè)注解,就可以將你的業(yè)務(wù)代碼通過切面的方式添加到事件的前后,而不需要修改事件的代碼
效果圖
如下圖所示,add方法內(nèi)并沒有調(diào)用其他的方法,但是其他方法仍然被執(zhí)行了。
只要給監(jiān)聽方法加@AddEventListener()注解就可以讓它在事件前后執(zhí)行了


監(jiān)聽原理
該方法是利用切面、注解、反射來實(shí)現(xiàn)SpringBoot的事件監(jiān)聽的
1.通過Aspect的切面,切入事件方法
首先使用Aspec的Around(也可以用before或者after,但是比較麻煩)注解,切入AddEvent的方法中,around注解的方法中,可以在事件方法的執(zhí)行前后添加業(yè)務(wù)代碼。但是我們不直接加入需要添加的業(yè)務(wù),進(jìn)入第二步驟。
2.利用反射獲取被AddEventAop注解的類和方法
利用反射Class.forName(class),獲取被AddEventAop注解的類(當(dāng)然你也可以修改一下,獲取所有的類),該類哪個(gè)方法被AddEventListener注解了,就執(zhí)行該方法,則監(jiān)聽執(zhí)行成功。
method.invoke(o, args);
注意(非常重要)
- AddEventListener使用的類上,必須被AddEventAop注解了,否則反射的時(shí)候方法不會(huì)被執(zhí)行。
- 事件的類必須是bean,否則切面失敗。
- 監(jiān)聽方法和(被監(jiān)聽方法)事件方法的參數(shù)數(shù)量,類型,順序必須一致,否則可能導(dǎo)致反射執(zhí)行方法失敗
核心源碼
@Around("@annotation(event)")
public Object addEventListener(ProceedingJoinPoint joinPoint, AddEventAop event) throws Throwable {
Object[] args = joinPoint.getArgs();
//存儲(chǔ)需要在方法執(zhí)行之后再執(zhí)行的類
List<Method> afterEventMethod = new ArrayList<>();
//反射獲取AddEventListener修飾的方法并執(zhí)行
//獲取自定義注解的配置
final Map<String, Object> beans = applicationContext.getBeansWithAnnotation(AddEventAop.class);
for (String key : beans.keySet()) {
//Spring 代理類導(dǎo)致Method無法獲取,這里使用AopUtils.getTargetClass()方法
Object o = beans.get(key);
Class<?> aClass = beans.get(key).getClass();
String name = aClass.getName();
//aop切面會(huì)導(dǎo)致方法注解丟失,在這里處理獲取原類名
if (name.contains("$$")){
String[] names = name.split("\\$\\$");
name=names[0];
aClass = Class.forName(name);
}
Method[] methods = aClass.getMethods();
for (Method method : methods) {
//獲取指定方法上的注解的屬性
AddEventListener annotation = method.getAnnotation(AddEventListener.class);
if (annotation!=null){
//執(zhí)行所有的注解了該類的方法
EventType value = annotation.value();
if (value.equals(EventType.BEFOREEVENT)){
method.invoke(o, args);
}else{
afterEventMethod.add(method);
}
}
}
}
//執(zhí)行被切面的方法
Object proceed = joinPoint.proceed(args);
//執(zhí)行需要在方法執(zhí)行之后再執(zhí)行的方法
for (Method method : afterEventMethod) {
Class<?> aClass = method.getDeclaringClass();
Object o = aClass.newInstance();
method.invoke(o, args);
}
return proceed;
}源碼地址
到此這篇關(guān)于SpringBoot利用切面注解及反射實(shí)現(xiàn)事件監(jiān)聽功能的文章就介紹到這了,更多相關(guān)SpringBoot事件監(jiān)聽內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- springboot 事件監(jiān)聽的實(shí)現(xiàn)方法
- SpringBoot Application事件監(jiān)聽的實(shí)現(xiàn)方案
- springboot+redis過期事件監(jiān)聽實(shí)現(xiàn)過程解析
- SpringBoot加載應(yīng)用事件監(jiān)聽器代碼實(shí)例
- SpringBoot監(jiān)聽事件和處理事件程序示例詳解
- SpringBoot ApplicationListener事件監(jiān)聽接口使用問題探究
- SpringBoot中的ApplicationListener事件監(jiān)聽器使用詳解
- Springboot事件監(jiān)聽與@Async注解詳解
- Java?Springboot異步執(zhí)行事件監(jiān)聽和處理實(shí)例
- SpringBoot實(shí)現(xiàn)事件監(jiān)聽(異步執(zhí)行)的示例代碼
相關(guān)文章
Java中的BufferedInputStream與BufferedOutputStream使用示例
BufferedInputStream和BufferedOutputStream分別繼承于FilterInputStream和FilterOutputStream,代表著緩沖區(qū)的輸入輸出,這里我們就來看一下Java中的BufferedInputStream與BufferedOutputStream使用示例:2016-06-06
Java實(shí)現(xiàn)解析ini文件對應(yīng)到JavaBean中
ini 文件是Initialization File的縮寫,即初始化文件,是windows的系統(tǒng)配置文件所采用的存儲(chǔ)格式。這篇文章主要介紹了通過Java實(shí)現(xiàn)解析ini文件對應(yīng)到JavaBean中,需要的可以參考一下2022-01-01
SpringCloud Feign遠(yuǎn)程調(diào)用與自定義配置詳解
Feign是Netflix公司開發(fā)的一個(gè)聲明式的REST調(diào)用客戶端; Ribbon負(fù)載均衡、 Hystrⅸ服務(wù)熔斷是我們Spring Cloud中進(jìn)行微服務(wù)開發(fā)非常基礎(chǔ)的組件,在使用的過程中我們也發(fā)現(xiàn)它們一般都是同時(shí)出現(xiàn)的,而且配置也都非常相似2022-11-11
Java流操作之?dāng)?shù)據(jù)流實(shí)例代碼
這篇文章主要介紹了Java流操作之?dāng)?shù)據(jù)流實(shí)例代碼,具有一定借鑒價(jià)值,需要的朋友可以參考下2018-01-01
Java 使用多線程調(diào)用類的靜態(tài)方法的示例
這篇文章主要介紹了Java 使用多線程調(diào)用類的靜態(tài)方法的示例,幫助大家更好的理解和使用Java,感興趣的朋友可以了解下2020-10-10
java webApp異步上傳圖片實(shí)現(xiàn)代碼
這篇文章主要為大家詳細(xì)介紹了java webApp異步上傳圖片實(shí)現(xiàn)代碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11
Java利用序列化實(shí)現(xiàn)對象深度clone的方法
這篇文章主要介紹了Java利用序列化實(shí)現(xiàn)對象深度clone的方法,實(shí)例分析了java序列化及對象克隆的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-07-07

