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的关系