Spring AOP 面向切面编程,简单来说:其实就是让关注点代码与业务代码分离。在原有的某个功能上增加些什么功能缺又不产生影响。
Join point(连接点):程序执行过程中的一个点,如方法的执行或异常的处理。在Spring AOP中,连接点总是表示方法的执行。通俗的讲,连接点即表示类里面可以被增强的方法。
Aspect(切面):Aspect切面表示Pointcut(切入点)和Advice(增强/通知)的结合。
Pointcut(切入点):执行目标对象方法,动态植入切面代码。可以通过切入点表达式,指定拦截哪些类的哪些方法; 给指定的类在运行的时候植入切面类代码。
Advice(增强/通知):指拦截到Joinpoint之后所要做的事情就是通知,通知分为前置通知、后置通知、异常通知、最终通知和环绕通知(切面要完成的功能)。
@Component
@Aspect
public class AspectDemo {
@Autowired
private MyService myService ;
//匹配com.aop.servicey.impl.MyServiceImpl类下的方法名以del开头、参数类型为Long的public方法
@Pointcut("execution(public * com.aop.service.impl.MyServiceImpl.del*(Long))")
public void conditionPointcut() {}
//使用matchCondition这个切入点进行增强,这里采用的是前置增强
@Before("conditionPointcut()")
public void before() {
System.out.println("before 前置通知......");
myService.look();
}
// @After后置通知 @AfterThrowing异常通知 @AfterReturning返回通知 @Around环绕通知
}
关于Pointcut(切入点)表达式网上非常多,有兴趣可以去了解一下。
Spring AOP的原理是代理模式。
代理(Proxy)是一种设计模式, 提供了对目标对象另外的访问方式;即通过代理访问目标对象。 这样好处: 可以在目标对象实现的基础上,增强额外的功能操作。(扩展目标对象的功能)。
就好比:卖房者(目标) —》 中介(代理) —》买房者
代理模式的关键点: 代理对象与目标对象。
代理模式又分为,静态代理/动态代理。
前提:代理对象,要实现与目标对象一样的接口。
优点:可以做到在不修改目标对象的功能前提下,对目标对象功能扩展。
缺点:因为实现与目标对象一样的接口,会产生很多的代理类。一旦目标接口添加或者减少方法,维护不便。
解决方法:采用动态代理或者代理工厂模式。
代理对象不需要实现接口,但是目标对象一定要实现接口,否则不能用动态代理。
代理对象,不需要实现接口。代理对象由JDK生成,动态的在内存中构建代理对象(需要我们指定创建 代理对象/目标对象 实现的接口的类型)。
Cglib代理,也叫做子类代理。在内存中构建一个子类对象从而实现对目标对象功能的扩展。
JDK的动态代理有一个限制,就是使用动态代理的对象必须实现一个或多个接口。如果想代理没有实现接口的类,就可以使用CGLIB实现。
CGLIB是一个强大的高性能的代码生成包,它可以在运行期扩展Java类与实现Java接口。它广泛的被许多AOP的框架使用,例如Spring AOP和dynaop,为他们提供方法的interception(拦截)。
CGLIB包的底层是通过使用一个小而快的字节码处理框架ASM,来转换字节码并生成新的类。不鼓励直接使用ASM,因为它要求你必须对JVM内部结构包括class文件的格式和指令集都很熟悉。
Cglib子类代理:
1) 需要引入cglib – jar文件, 但是spring的核心包中已经包括了cglib功能,所以直接引入spring-core-3.2.5.jar即可。
2)引入功能包后,就可以在内存中动态构建子类
3)代理的类不能为final, 否则报错。
4) 目标对象的方法如果为final/static, 那么就不会被拦截,即不会执行目标对象额外的业务方法。
在Spring的AOP编程中,如果加入容器的目标对象有实现接口,用JDK代理;如果目标对象没有实现接口,用Cglib代理;
1.集中处理某一关注点/横切逻辑。
2.可以很方便的添加/删除关注点。
3.侵入性少,增强代码可读性及可维护性。
4.代码的耦合减少,没有多余的代码。