基础概念
Spring aop兼容了aspect j的注解,不过实现原理完全不同哦
Spring是一个容器,凡是在容器里的对象才会有Spring所提供的这些服务和功能。 AOP也不例外。Spring配置的那些Bean经过 载入 、解析 和 注册 这三个过程后,在框架内部被抽象封装成BeanDefinition这种类型,最终所有的BeanDefinition交由BeanFactory当中的definitionMap统一管理起来。
在getbean的时候,spring框架会完成依赖注入。Spring当中提供了两种实例化方案: BeanUtils 和 Cglib 方式。BeanUtils实现机制是通过Java的反射机制,Cglib是一个第三方类库采用的是一种字节码加强方式机制。 Spring中采用的默认实例化策略是Cglib。可以在spring配置文件中修改如下配置
<aop:aspectj-autoproxy proxy-target-class="true"/>
一个简单的DEMO
定义一个接口
public interface Person {
String sayHello(String name);
}
实现类
@Component
public class Chinese implements Person {
@Timer
@Override
public String sayHello(String name) {
System.out.println("-- sayHello() --");
return name + " hello, AOP";
}
public void eat(String food) {
System.out.println("我正在吃:" + food);
}
}
这里的@Timer注解是我自己定义的一个普通注解,用来标记Pointcut。
定义Aspect
@Aspect
@Component
public class AdviceTest {
@Pointcut("@annotation(com.listenzhangbin.aop.Timer)")
public void pointcut() {
}
@Before("pointcut()")
public void before() {
System.out.println("before");
}
}
实际使用
@SpringBootApplication
@RestController
public class SpringBootDemoApplication {
//这里必须使用Person接口做注入
@Autowired
private Person chinese;
@RequestMapping("/test")
public void test() {
chinese.sayHello("listen");
System.out.println(chinese.getClass());
}
public static void main(String[] args) {
SpringApplication.run(SpringBootDemoApplication.class, args);
}
}
输出
before
-- sayHello() --
class com.sun.proxy.$Proxy53
生成了一个匿名的代理类。
如果被代理类没有实现接口(或者要调用没有声明为接口的public方法),则输出如下
before
-- sayHello() --
class com.listenzhangbin.aop.Chinese$$EnhancerBySpringCGLIB$$56b89168
可以看出,此时使用的CGLIB的增强方案。
Spring AOP原理
Spring 代理选择方法如下:
1.当被代理对象,实现了至少一个接口时会使用JDK动态代理.(所有接口都会被代理,但是你只能以接口类型从spring中获取bean)。
2.当被代理对象没有实现任何接口时,spring会使用CGLIB.(要将CGLIB的二进制发行包放在classpath下)
关于java中的代理,请看
https://blog.csdn.net/define_us/article/details/78682396
AOP中的异常
注意Spring AOP的框架的异常特殊性。直接抛出的Exception会被封装为java.lang.reflect.UndeclaredThrowableException。该异常的异常的message是null。
解决办法之一,使用e.getCause().getMessage()获取详细异常信息
解决办法之二,抛出RUMTIME EXCEPTION