Spring?代理?Bean?獲取不到原始?Bean?對象注解解決方法
一、問題描述
在接受 mq 消息的時(shí)候,需要做一個(gè)重試次數(shù)限制,如果超過 maxNum
就發(fā)郵件告警,不再重試。 所以我需要對 consumer
對象進(jìn)行代理,然后如果超過異常次數(shù),我直接返回成功,并且發(fā)送成功消息,但是我獲取 consumer handler
方法的方式是通過 method.getAnnotation(XXClient.class)
方式,那么就會(huì)返回 null。
1.問題示例代碼
- 目標(biāo)類, 我這里就之定義一個(gè)
test
方法,里面做一些個(gè)簡單的打印。
@Component public class TestBean { @Anno public void test() { System.out.println("test ....."); } }
- 代理邏輯邏輯處理, 主要就是做一個(gè)
@Around
的方法覆蓋,保證在調(diào)用目標(biāo)方法之前,先輸出我插入的邏輯。
@Target({ElementType.METHOD}) @Retention(RetentionPolicy.RUNTIME) public @interface Anno { String key() default "100%"; } @Aspect @Component public class AnnoAspect { @Around("@annotation(anno)") public Object anno(ProceedingJoinPoint point, Anno anno) throws Throwable { System.out.println("anno invoke!!!!!!"); return point.proceed(); } }
- 調(diào)用點(diǎn), 通過
AnnotationConfigApplicationContext
獲取bean
. 然后通過getMethods()
獲取所有的方法,最后查找Anno
注解的Method
對象。
AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(BeanStart.class); TestBean bean = applicationContext.getBean(TestBean.class); Class<? extends TestBean> classz = bean.getClass(); Method[] methods = classz.getMethods(); for (Method m : methods) { Anno annotation = m.getAnnotation(Anno.class); if (annotation != null) { System.out.println(" ============= invoke test ==========="); m.invoke(bean, new Object()); } }
由于 m.getAnnotaion(Anno.class)
無法獲取到注解信息,所以執(zhí)行 test
方法失敗, 到此問題還原完畢,我們再來看看如何解決。
二、解決方案
通過 Anno ao = AnnotationUtils.findAnnotation(method, Anno.class);
方法獲取即可。
有的代碼是這樣寫的 :
String name = classz.getName(); boolean isSpringProxy = name.indexOf("SpringCGLIB$$") >= 0; Method[] methods; if (isSpringProxy) { methods = ReflectionUtils.getAllDeclaredMethods(AopUtils.getTargetClass(bean)); } else { methods = classz.getMethods(); } // 省略部分代碼 if (isSpringProxy) { annotation = AnnotationUtils.findAnnotation(method, MqClient.class); } else { annotation = method.getAnnotation(Anno.class); }
這里他會(huì)做一個(gè)判斷,如果是代理對象就調(diào)用 ReflectionUtils.getAllDeclaredMethods
獲取所有的方法, 然后再去拿注解的時(shí)候二次判斷一下,如果存在代理,那么就通過 AnnotationUtils.findAnnotation
感覺是相當(dāng)?shù)膰?yán)謹(jǐn)。
總結(jié):
Spring 提供了非常強(qiáng)大的一站式開發(fā)功能,而且還提供了比較優(yōu)秀的工具方法比如: BeanUtils
、ReflectionUtils
、AnnotationUtils
等,這些都是我們值得掌握的基礎(chǔ)工具類。
相關(guān)文章
在logback.xml中自定義動(dòng)態(tài)屬性的方法
這篇文章主要介紹了在logback.xml中自定義動(dòng)態(tài)屬性的方法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2019-08-08JAVA獲取當(dāng)前項(xiàng)目和文件所在路徑的實(shí)例代碼
這篇文章主要介紹了JAVA獲取當(dāng)前項(xiàng)目和文件所在路徑的實(shí)例代碼,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-03-03Java 14 發(fā)布了,你還會(huì)使用Lombok?
2020年3月17日發(fā)布,Java正式發(fā)布了JDK 14 ,目前已經(jīng)可以開放下載。在JDK 14中,共有16個(gè)新特性,本文主要來介紹其中的一個(gè)特性:JEP 359: Records,需要的朋友可以參考下2020-04-04Java數(shù)據(jù)庫連接池的幾種配置方法(以MySQL數(shù)據(jù)庫為例)
這篇文章主要介紹了Java數(shù)據(jù)庫連接池的幾種配置方法(以MySQL數(shù)據(jù)庫為例) 的相關(guān)資料,需要的朋友可以參考下2016-07-07Ajax 驗(yàn)證用戶輸入的驗(yàn)證碼是否與隨機(jī)生成的一致
本文主要介紹了后臺(tái)驗(yàn)證用戶輸入的驗(yàn)證碼是否與隨機(jī)生成的驗(yàn)證碼一致的方法,具有很好的參考價(jià)值,下面跟著小編一起來看下吧2017-02-02