SpringBoot利用切面注解及反射實(shí)現(xiàn)事件監(jiān)聽功能
前言
當(dāng)某個事件需要被監(jiān)聽的時候,我們需要去做其他的事前,最簡單的方式就是將自己的業(yè)務(wù) 方法追加到該事件之后。
但是當(dāng)有N多個這樣的需求的時候我們都這樣一個個去添加修改事件的源碼嗎?
這篇文章將告訴你如何用一個注解,就可以將你的業(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)然你也可以修改一下,獲取所有的類),該類哪個方法被AddEventListener注解了,就執(zhí)行該方法,則監(jiān)聽執(zhí)行成功。
method.invoke(o, args);
注意(非常重要)
- AddEventListener使用的類上,必須被AddEventAop注解了,否則反射的時候方法不會被執(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(); //存儲需要在方法執(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切面會導(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)文章
SpringBoot如何返回頁面的實(shí)現(xiàn)方法
SpringBoot中使用Controller和頁面的結(jié)合能夠很好地實(shí)現(xiàn)用戶的功能及頁面數(shù)據(jù)的傳遞。本文介紹了如何實(shí)現(xiàn)頁面的返回以及這里面所包含的坑,感興趣的可以了解一下2021-07-07Spring之ShutDown?Hook死鎖現(xiàn)象解讀
這篇文章主要介紹了Spring之ShutDown?Hook死鎖現(xiàn)象解讀,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2023-04-04Java中使用BigDecimal進(jìn)行精確運(yùn)算
這篇文章主要介紹了Java中使用BigDecimal進(jìn)行精確運(yùn)算的方法,非常不錯,需要的朋友參考下2017-02-02rabbitmq使用springboot實(shí)現(xiàn)direct模式(最新推薦)
這篇文章主要介紹了rabbitmq使用springboot實(shí)現(xiàn)direct模式,本文通過示例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-07-07