个人博客原文地址:http://www.ltang.me/2016/08/18/get-annotation-aop-proxy/
背景描述
自定义了一个注解:
@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface ScheduledTask {
}
在一个实现类上添加此注解:
@ScheduledTask
public class TimerTaskServiceImpl implements TimeTaskService {
在代码中期望通过判断是否包含注解来执行操作:
if(bean.getClass().isAnnotationPresent(ScheduledTask.class)){
...
本来一切运行正常。但是,当我试图在实现类上添加事务注解@Transactional后,这里的判断就一直是false。调试发现,当跑到这一行代码时,bean不是实现类,而是一个Proxy代理类(JdkDynamicAopProxy),所以自定义的注解一直获取不到。搜索之后,才知道这是spring的事务管理对添加了@Transactional注解的类做了其他操作,我们获取到的是一个处理后的代理类。
解决方案
于是试图使用下面的代码获取到实际的类:
Object target = ((SingletonTargetSource)((Advised) processor.getIface()).getTargetSource()).getTarget();
或者简单点:
Class<?> = ((Advised) processor.getIface()).getTargetClass();
然而由于框架里面ClassLoader不一致的问题,导致不能直接转换,所以只能通过反射的方式获取到实际类:
long count = new ArrayList<>(Arrays.asList(processor.getIface().getClass().getInterfaces()))
.stream()
.filter(m -> m.getName().equals("org.springframework.aop.framework.Advised"))
.count();
Class<?> ifaceClass = (Class) (count > 0 ? processor.getIface().getClass().getMethod("getTargetClass").invoke(processor.getIface()) : processor.getIface().getClass());
if(ifaceClass.isAnnotationPresent(ScheduledTask.class)){
...
解决了这个问题。仅作记录。
本文介绍了解决在Spring框架中使用自定义注解并结合AOP和事务管理时遇到的问题。当一个类同时被自定义注解和@Transactional注解修饰时,由于Spring的AOP代理机制,直接通过bean获取自定义注解会失败。文章提供了通过反射获取实际目标类的方法,从而能够正确判断自定义注解的存在。
1808

被折叠的 条评论
为什么被折叠?



