Spring AOP分析(1) -- 基本概念

时间:2024-07-17 22:36:32

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可以看作两步:

  1. 根据配置文件或者注解生成代理对象
  2. 执行代理对象,代理对象中包括增强方法和被代理对象

下面简单了解下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的具体细节