问题描述
我在接受 mq 消息的时候,需要做一个重试次数限制,如果超过 maxNum
就发邮件告警,不再重试。 所以我需要对 consumer
对象进行代理,然后如果超过异常次数,我直接返回成功,并且发送成功消息,但是我获取 consumer handler
方法的方式是通过 method.getAnnotation(XXClient.class)
方式,那么就会返回 null。
问题示例代码
- 目标类, 我这里就之定义一个
test
方法,里面做一些个简单的打印。
@Component public class TestBean { @Anno public void test() { System.out.println("test ....."); } }
- 代理逻辑逻辑处理, 主要就是做一个
@Around
的方法覆盖,保证在调用目标方法之前,先输出我插入的逻辑。
@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(); } }
- 调用点, 通过
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)
无法获取到注解信息,所以执行 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); }
这里他会做一个判断,如果是代理对象就调用
ReflectionUtils.getAllDeclaredMethods
获取所有的方法, 然后再去拿注解的时候二次判断一下,如果存在代理,那么就通过 AnnotationUtils.findAnnotation
感觉是相当的严谨。
总结:Spring 提供了非常强大的一站式开发功能,而且还提供了比较优秀的工具方法比如: BeanUtils
、ReflectionUtils
、AnnotationUtils
等,这些都是我们值得掌握的基础工具类。