spring aop final 方法

最近做项目用到了spring aop,由于在项目里面用到了模板方法模式,在抽象类里面有一个final 方法。controller 调用这个final 方法的时候报空指针异常,一般我们都知道使用 aop 在controller 中注入的对象是spring 创建的代理对象,默认使用cglib。这里我们称代理对象为 proxy,称被代理的对象是target。在debuge的过程中发现,spring 注入到controller的proxy 中的所有属性都是null,包括日志logger都是null,即使 我写了 private String kk=“1111”,这个 kk 属性在proxy 中也是null。在本次中由于调用了proxy中的方法所以报了空指针异常。可是我在用aop定义切面的时候用的是 @annotation 定义的,按道理说只会拦截某一个方法,其他的方法应该直接调用 target 中相应的方法,主要是这个方法是 final 修饰的。由于默认使用cglib 来生成proxy,这个proxy就是 target的子类。而对于 final 方法子类是不能覆盖的,走的代码流程依然是 target 里面的。这是最坑的一个地方,我在网上找了很多资料,绝大多数都是说,对于没有final 修饰的方法是直接调用proxy的发方法,对于有 final 修饰的方法是调用target方法。按照这种想法proxy和target只是一个实例。类的结构类似这种
spring aop final 方法_第1张图片
Demo 是 target DemoSub 是 proxy。我们只有一个实例就是 DemoSub。在调用DemoSub ,由于 kkk 方法用final 修饰了,DemoSub 不能覆盖。所以当调用kkk方法的时候实际是调用Demo的代码。***我觉得这是不正确的。***。主要是实例个数的问题。如下
spring aop final 方法_第2张图片
最终生成的是两个实例,在proxy中有一个属性关联target。我们注入的proxy对象,如果是final 方法,直接就走 proxy的kkk方法,这个时候proxy中所有的属性是没有赋值都是null,其中proxy中也有一个隐式指针super执行其父类,但这个父类也是没有被spring 容器处理过的。比如里面的autowrite 属性都是null,如果不是final 修饰的方法,proxy中会先查看当前方法是否被定义为拦截的,如果是要拦截的则需要先执行 @Before 或者 @Around 注释的方法。如果不是则用 proxy中关联的 target 直接invoke 处理,这个target 才是我们真正的对象,里面所有的属性都是都是被处理过的。所有的aware接口也是被调用过的。
总结一下主要是生成了几个对象,按照网上说的之生成了一个对象,用supper调用,可实际上有两个,proxy中有个属性和target做关联了。
后面还有一个问题。切面逻辑无法被执行到。还是上面的例子去掉final
spring aop final 方法_第3张图片
在这里我拦截的方法是syahi而我controller中调用的是 kkk 方法。这个时候对syahi方法的拦截逻辑是无法被执行的。主要是在controller中注入的是proxy 调用proxy的kkk方法,proxy发现这个方法没有被拦截,所以直接用 target 执行,这个时候执行kkk方法的对象变成了target,这里的this 指向是 target对象。在kkk方法里面调用 syahi 方法也就只是调用 target 的syahi 方法,想要拦截逻辑被执行这里调用syahi方法的对象不能是 target 必须是 proxy。spring aop中可以处理源码如下
spring aop final 方法_第4张图片
有一个AopContext,应该通过 ThreadLocal 。在kkk方法改动如下

    public void kkk(){
        ((Demo) AopContext.currentProxy()).syahi();
        System.out.println("kkkk");
    }

即可。在这里记录一下。

你可能感兴趣的:(spring,boot)