spring aop获取目标对象的方法对象(续)

来源:互联网 发布:手机翻译软件排行 编辑:程序博客网 时间:2024/05/17 01:04

    在上一篇文章中,判断目标方法上是否有注解Permission,说使用如下方式

Signature signature = pjp.getSignature();    MethodSignature methodSignature = (MethodSignature)signature;    Method targetMethod = methodSignature.getMethod();            Class clazz = targetMethod.getClass();    if(clazz.isAnnotationPresent(Permission.class)){     ...}
clazz.isAnnotationPresent(Permission.class)
方法返回false,然后才出了上一篇博文,博文中到最后还有个矛盾没解决,原来是这边判断方法上是否有注解出了问题,鄙视下自己鄙视

    通过ProceedingJoinPoint的getSignature()方法获取Signature,然后转型成MethodSignature,然后通过MethodSignature的getMethod()方法,在目标对象没有实现接口情况下(使用cglib生成代理),getMethod()获取的Method对象其实已经是目标对象上的Method,而不是代理对象上的Method对象,至于没有获取到method对象上的Permission注解,是因为判断的时候使用的对象不对,原来判断方式是这样的:

Class clazz = targetMethod.getClass();    if(clazz.isAnnotationPresent(Permission.class)){     ...}
这边又去获取了targetMethod的字节码对象,获取到得是Method类,而不是代表拦截到的目标对象的实际detail(...)这个方法Method的一个实例,这边判断是否有注解当然就返回false了,为了进一步验证

Signature signature = pjp.getSignature();    MethodSignature methodSignature = (MethodSignature)signature;    Method targetMethod = methodSignature.getMethod();  
获取的targetMethod是目标对象上的而非代理对象上的,可以在代码中增加如下输出:

Signature signature = pjp.getSignature();MethodSignature methodSignature = (MethodSignature)signature;Method targetMethod = methodSignature.getMethod();System.out.println("target:" + targetMethod.getDeclaringClass().getName());System.out.println("targetsuper:" + targetMethod.getDeclaringClass().getSuperclass().getName());System.out.println("proxy:" + pjp.getThis().getClass().getName());System.out.println("proxysuper:" + pjp.getThis().getClass().getSuperclass().getName());
测试结果:

target:com.jiangnan.cms.controller.UserControllertargetsuper:java.lang.Objectproxy:com.jiangnan.cms.controller.UserController$$EnhancerByCGLIB$$84064edfproxysuper:com.jiangnan.cms.controller.UserController
其中
targetMethod.getDeclaringClass()
取到的是定义这个方法的类或接口,可以理解为该方法所在的类或接口,从打印的结果可以看出targetMethod是目标对象身上的,因为其getDeclaringClass().getSuperclass().getName()是Object,如果是代理对象身上的,根据cglib生成代理对象的原则(继承目标对象,重写父类非final方法),getDeclaringClass().getSuperclass().getName()获取到得应该是其父类即com.jiangnan.cms.controller.UserController,而非java.lang.Object。

    至此上一篇博文中的矛盾已经解决。