Spring AOP,AspectJ, CGLIB 有点晕
Spring AOP 实现原理与 CGLIB 应用
hello.java public class SayHelloService { public void say(){ System.out.print("Hello AspectJ"); } } LogAspect.java public aspect LogAspect{ pointcut logPointcut():execution(void SayHelloService.say()); after():logPointcut(){ System.out.println("记录日志 ..."); } }
运行时
ajc -d . SayHelloService.java LogAspect.java // 生成 SayHelloService.class java SayHelloService
javac
也就是, 除非用上了 ajc
去编译 java 源代码,否则,都是spring aop + jdk proxy/cglib 那一套。
@Component public class SayHelloService { public void say(){ System.out.print("Hello AspectJ"); } } @Aspect @Component public class LogAspect { @After("execution(* com.ywsc.fenfenzhong.aspectj.learn.SayHelloService.*(..))") public void log(){ System.out.println("记录日志 ..."); } }
Spring的两种代理JDK和CGLIB的区别浅谈
java动态代理是利用反射机制生成一个实现代理接口的匿名类,在调用具体方法前调用InvokeHandler来处理。
而cglib动态代理是利用asm开源包,对代理对象类的class文件加载进来,通过修改其字节码生成子类来处理。
1、如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP 2、如果目标对象实现了接口,可以强制使用CGLIB实现AOP 3、如果目标对象没有实现了接口,必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换
Spring AOP无效分析
个人发现的一个情况:
Spring + Aspectj annotation, 假设存在
interface IHelloService { void hello(String name); } class HelloService implements IHelloService { public void hello(String name){ System.out.pringln("hello " + name); } }
使用注解@LogTime 将方法执行时间加入到 线程的threadlocal 中
注解加在接口上 | 注解加在类上 | |
---|---|---|
<aop:aspectj-autoproxy/>
|
有效 | 无效 |
<aop:aspectj-autoproxy proxy-target-class="true"/>
|
有效 | 有效 |
理论上,刨除性能、必须实现接口的因素, cglib 和 jdk proxy 是可以等效替换的,现在却出现了这个情况。
<aop:aspectj-autoproxy/>
时
ProceedingJoinPoint pjp
无法获取 HelloService.hello
上的注解。但可以获取 IHelloService.hello
上的注解 接口方法上的注解无法被 @Aspect 声明的切面拦截的原因分析 (待进一步明确描述)
Spring中DispacherServlet、WebApplicationContext、ServletContext的关系