AOP全称是Aspect Oriented Programming,面向切面编程,是面向对象编程(OOP:Object Oriented Programming)的补充和完善。一般在系统中,OOP利用继承、封装、多态来模拟处理业务模块和逻辑;AOP则是封装非业务但又被各业务模块频繁调用的功能,如日志、权限认证、事物。利用AOP可有效减少系统中的重复代码、降低模块间的耦合、利于维护。
基础概念
-
切面(Aspect)
切面是对横切关注点的抽象,我们对哪些方法进行拦截,拦截后怎么处理,这个定义为切面。如Spring中的事务管理
-
连接点(Joinpoint)
通俗来讲就是在哪些地方可以被拦截。Spring中连接点是方法级别的,但实际中可以是字段、可以是构造器
-
切入点(pointcut)
简单讲就是被拦截的连接点,也即是后续添加通知的位置
-
目标对象(Target Object)
切入点所在的类,也即是被拦截方法所属类
-
通知(Advice)
在切入点切入的操作,也即是我们要对被代理对象所加入的操作。按照类型划分主要有五种,前置(BeforeAdvice),后置( AfterReturningAdvice),环绕(AroundAdvice),最终(AfterFinallyAdvice),异常(AfterThrowringAdvice)
-
引入(Introduction)
添加方法或字段到被代理的对象
-
织入(Weaving)
重新创建一个代理对象,代理对象中包括通知和被代理对象
核心接口类
整个Spring AOP可以看作两步:
- 根据配置文件或者注解生成代理对象
- 执行代理对象,代理对象中包括增强方法和被代理对象
下面简单了解下Spring AOP中核心的接口类
InvocationHandler
接口,从字面不难理解是调用处理器的意思,该接口是一个祖宗级接口,大多动态代理接口都会实现它,如JdkDynamicAopProxy。其中只定义的了一个方法
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable;
AopProxy
接口,也是一个祖宗级接口,定义代理。只有三个实现类CglibAopProxy、JdkDynamicAopProxy、ObjenesisCglibAopProxy,接口内部只定义了两个方法
Object getProxy();
Object getProxy(ClassLoader classLoader);
AopProxyFactory
接口,生成代理的工厂。接口只有一个方法,定义了创建代理的方法,实现如下所示:
public interface AopProxyFactory {
AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException;
}
在Spring AOP中,该接口只有一个实现类DefaultAopProxyFactory。DefaultAopProxyFactory实现了接口中的createAopProxy方法,定义了何时使用JdkDynamicAopProxy、何时使用ObjenesisCglibAopProxy代理。源码如下:
public class DefaultAopProxyFactory implements AopProxyFactory, Serializable {
@Override
public AopProxy createAopProxy(AdvisedSupport config) throws AopConfigException {
if (config.isOptimize() || config.isProxyTargetClass() || hasNoUserSuppliedProxyInterfaces(config)) {
Class<?> targetClass = config.getTargetClass();
if (targetClass == null) {
throw new AopConfigException("TargetSource cannot determine target class: " +
"Either an interface or a target is required for proxy creation.");
}
if (targetClass.isInterface() || Proxy.isProxyClass(targetClass)) {
return new JdkDynamicAopProxy(config);
}
return new ObjenesisCglibAopProxy(config);
}
else {
return new JdkDynamicAopProxy(config);
}
}
private boolean hasNoUserSuppliedProxyInterfaces(AdvisedSupport config) {
Class<?>[] ifcs = config.getProxiedInterfaces();
return (ifcs.length == 0 || (ifcs.length == 1 && SpringProxy.class.isAssignableFrom(ifcs[0])));
}
}
从实现中可以看出Spring AOP中默认是应用JDK动态代理实现AOP功能,当代理对象为接口时使用JDK动态代理,否则使用ObjenesisCglibAopProxy实现。
Advised、Pointcut等接口分别定义了通知和切入点。
本小结简单了解了基础概念、分析了核心的祖宗级接口类。下一节将重点探讨下JDK动态代理和CBLIB实现AOP的具体细节