美文网首页技术干货程序员
Spring技术内幕(二):AOP(一)

Spring技术内幕(二):AOP(一)

作者: WhyNotYue | 来源:发表于2017-04-08 09:35 被阅读0次

    一,概述


    1. 原理:面向切面编程

    从关注点中分离出横切关注点,通俗点讲,即将解决特定领域问题的代码从业务逻辑中脱离出来。举个栗子,简单的增删改中一般有事务管理,但事务管理与业务逻辑无关,是一个特定重复的操作,这个时候,我们就可以把事务管理从增删改的业务逻辑中脱离出来。在这里,我们的关注点就是增删改等业务逻辑,横切关注点就是事务管理。


    2.1 Advice通知

    1. 作用:
    • 描述Spring AOP围绕方法调用而注入的切面行为。
    1. 类型(这里只列举三种)
    • BeforeAdvice:在目标实例的方法(即,JoinPoint:连接点)执行前嵌入执行额外功能。
    • AfterAdvice:在目标实例的方法执行后嵌入执行额外功能。
    • ThrowAdvice:在目标实例的方法抛出异常时执行Advice。

    2.2 Pointcut切点

    决定Advice通知应该作用于哪个连接点。


    2.3 Advisor通知器

    用来结合Advice和Pointcut


    二,Spring AOP实现的核心技术


    动态代理

    这里以JVM的动态代理模式举例

    • 图解代理模式:


      JVM动态代理.png

      如图3-9,代理对象Proxy与目标对象RealSubject实现了同一个接口Subject。当调用目标对象的request()方法时,往往会被代理对象拦截。通过这种拦截, 为目标对象的方法操作做了铺垫,所以称之为代理模式。


    三,建立AopProxy代理对象


    1. 设计原理

    ProxyFactory相关的类继承关系:


    Proxy类继承关系.png
    • ProxyConfig:数据基类,为子类提供配置属性
    • AdvisedSupport:封装了AOP对通知和通知器的相关操作
    • ProxyCreatorSupport:子类创建AOP代理对象的辅助类,继承以上提到的基类的功能实现
    • AspectJProxyFactory:具体的AOP代理对象的生成。集成Spring和AspectJ。
    • ProxyFactory:具体的AOP代理对象的生成。为使用Spring AOP的应用提供AOP功能的封装。需要编程式使用Spring AOP功能。
    • ProxyFactoryBean:具体的AOP代理对象的生成。为使用Spring AOP的应用提供AOP功能的封装。可以在IOC容器中完成声明式配置。

    2. 配置ProxyFactoryBean

    <!--配置ProxyFactoryBean-->
        <!--1,定义使用的通知器Advisor-->
        <bean id="testAdvisor" class="com.abc.TestAdvisor"/>
        <!--2,定义ProxyFactoryBean-->
        <bean id="testAOP" class="org.springframework.aop.framework.ProxyFactoryBean"/>
        <!--3,设定与AOP实现相关的重要属性-->
        <!--3.1,proxyInterface-->
        <property name="proxyInterfaces"><value>com.test.AbcInterface</value></property>
        <!--3.2,target(目标对象)-->
        <property name="target">
            <bean class="com.abc.TestTarget"/>
        </property>
        <!--3.3,interceptorNames(需要定义的通知器)-->
        <property name="interceptorNames">
            <list><value>testAdvisor</value></list>
        </property>
    

    3. ProxyFactoryBean生成AopProxy代理对象

    AopProxy的生成过程:两种方式,依赖JDK或者CGLIB提供的Proxy特性。


    AopProxy的生成过程.png
    • initializeAdvisorChain():初始化通知器链,通知器链中封装了一系列从配置中读取的拦截器,为代理对象的生成做好准备。
    • getSingletonInstance():生成Singleton类型的Proxy
    • DefaultAopProxyFactory:AopProxy接口类,AopProxy有两个子类实现,一个是JdkDynamicAopProxy,一个是CglibProxyFactory

    4. JDK生成AopProxy代理对象

    // JdkDynamicAopProxy生成Proxy代理对象
        @Override
        public Object getProxy(ClassLoader classLoader) {
            if (logger.isDebugEnabled()) {
                logger.debug("Creating JDK dynamic proxy: target source is " + this.advised.getTargetSource());
            }
            // 1,从advised对象中取得代理对象的代理接口配置
            Class<?>[] proxiedInterfaces = AopProxyUtils.completeProxiedInterfaces(this.advised, true);
            findDefinedEqualsAndHashCodeMethods(proxiedInterfaces);
            // 2,调用JDK生成Proxy
            return Proxy.newProxyInstance(classLoader, proxiedInterfaces, this);
        }
    
    • newProxyInstance方法:需要指明3个参数,类装载器,代理接口,Proxy回调方法所在的对象,这个对象要实现InvocationHandler接口
    • InvocationHandler接口: 反射类接口,定义了invoke方法,提供代理对象的回调入口。

    5. CGLIB生成AopProxy代理对象

    Cglib2AopProxy生成AopProxy代理对象


    CGLIB生成Proxy代理对象.png

    配置Enhancer对象,通过Enhancer对象的callback回调设置生成代理对象。其中通过设置DynamicAdvisedInterceptor拦截器来完成AOP功能的。


    参考资料:Spring技术内幕(第2版) 计文柯

    相关文章

      网友评论

        本文标题:Spring技术内幕(二):AOP(一)

        本文链接:https://www.haomeiwen.com/subject/yasaottx.html