Spring学习笔记 —— AOP(面向切面编程) 之使用ProxyFactoryBean实现AOP

引言

到上一篇文章Spring学习笔记 —— Spring Context为止,我们已经基本了解Spring中的基本对象——Bean——的创建、相关属性的注入以及获取。其实在这不难发现,Spring的容器设计与Java的对象设计之间是有相似的地方的,比如BeanDefinition和Class对象,Bean和Object对象。

而Java的反射,则对应到了Spring中的AOP(面向切面编程),当然,AOP有着比反射更强大的功能以及更方便的配置。关于反射,我在前面已经有系列文章分析过了,有兴趣的可以从这里开始再阅读一下。

Spring中的AOP分为两种,一种是由Spring框架实现,基于ProxyFactoryBean的AOP,而另外一种则是基于AspectJ的AOP,我们首先介绍前者,在后续的文章也会介绍后者。

本文主要分成三部分,AOP概念的介绍,Spring AOP实例以及Spring AOP具体实现的分析。

AOP(Aspect Oriented Programming)简介

关于什么是AOP,我想通过一个《Spring in Action》中的一个例子来解释。

我们每家每户都会有电脑,电脑会耗电,而每家每户都会有一个电表来记录用电量,每个月会有人来查电表,这样电力公司就知道该收取多少电费了。

但是,如果没有电表,也没有人来查看用电量,而是由户主来联系电力公司并报告自己的用电量。虽然可能会有一些户主会很详细地记录所有的用电量,但是肯定大多数人并不会这么做。因为每个人每天都有很多事要处理,而监控电力使用情况会浪费他们大量时间,不交电费对他们只有好处而没有坏处。

软件系统的某些功能就像我们家里的电表,这些功能需要应用到系统的多个地方,但却不适合在应用到的地方被显式调用。

监控电量是一个很重要的功能,但却并不是大多数家庭重点关注的问题。监控电量更像是一个被动事件。

在软件中,有些行为对于大多数应用都是通用的。日志、安全和事务管理的确很重要。但它们是否应用对象主动参与的行为呢?如果让应用对象只关注于自己所针对的业务领域问题,而其他方面的问题由其他应用对象来处理,会不会更好呢?

在软件开发中,分布于应用中多处的功能被称为横切关注点(cross-cutting concerns)。通常,这些横切关注点从概念上是与应用的业务逻辑相分离的(但是往往直接嵌入到应用的业务逻辑之中)。将这些横切关注点与业务逻辑相分离正式面向切面编程(AOP)所要解决的。

而在AOP中,会涉及到一些新的概念,现在先对这些概念进行描述。

名称含义
切面(Aspect)作为提供业务代码之外的功能对象,如例子中提到的电表,记录了每家每户的用电量。程序中有可能会是日志,输出特定的日志结果。
目标对象(target object)执行业务逻辑的对象,如例子中每个独立的户主。
织入(Weaving)就是将切面中的对象与目标对象糅合在一起的过程。
连结点(JoinPoint)需要插入切面的地方,通常指业务代码中具体某个对象的某个方法。而在例子中,就是需要装电表的家庭(有些地方可能不需要安装电表)。
通知(Advice)通知是指在连接点上发生的事情,比如说电表会记录用电量。
切点(pointCut)是指一系列连结点的组合

在Spring AOP中暂时不会用到切面的概念,但是在AspectJ AOP中则会用到。下面我们就实现一个Spring AOP吧。

Spring AOP实例

首先是TargetObject,TargetObjectSample.java

public class TargetObjectSample {
    public void getMessage() {
        System.out.println("getMsg!");
    }

    public void getPointCutMessage() {
        System.out.println("get point cut Msg!");
    }
}

再声明一个连接点。PointCutSample.java

public class PointCutSample implements Pointcut{

    @Override
    public ClassFilter getClassFilter() {
        return ClassFilter.TRUE;
    }

    @Override
    public MethodMatcher getMethodMatcher() {
        return new MethodMatcher(){

            @Override
            public boolean matches(Method method, Class<?> targetClass) {
                if(method.getName().contains("PointCut")){
                    return true;
                }
                return false;
            }

            @Override
            public boolean isRuntime() {
                return true;
            }

            @Override
            public boolean matches(Method method, Class<?> targetClass, Object... args) {
                if(method.getName().contains("PointCut")){
                    return true;
                }
                return false;
            }};
    }

}

还有AdvisorSample.java

public class AdvisorSample implements PointcutAdvisor{
    @Override
    public Advice getAdvice() {
        return new BeforeAdviceSample();
    }

    @Override
    public boolean isPerInstance() {
        return false;
    }

    @Override
    public Pointcut getPointcut() {
        return new PointCutSample();
    }
}

最后是AdviceBeforeAdviceSample.java

public class BeforeAdviceSample implements MethodBeforeAdvice{

    @Override
    public void before(Method method, Object[] args, Object target) throws Throwable {
        System.out.println("before method execution!");
    }

}

配置文件beans.xml

<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
    http://www.springframework.org/schema/beans/spring-beans-3.0.xsd
    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context-3.0.xsd">

    <bean id="advisorSample" class="com.study.Spring.AdvisorSample"></bean>

    <bean id="targetSample" class="com.study.Spring.TargetObjectSample"></bean>

    <bean id="aopSample" class="org.springframework.aop.framework.ProxyFactoryBean">
       <property name="targetName">
            <value>targetSample</value>
        </property>
        <property name="interceptorNames">
            <list>
                <value>advisorSample</value>
            </list>
        </property>
    </bean>
</beans>

Maven 依赖

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>4.2.6.RELEASE</version>
    </dependency>

主函数App.java

public class App 
{
    public static void main( String[] args ) {
        ApplicationContext app =  new ClassPathXmlApplicationContext("beans.xml");

        TargetObjectSample obj = (TargetObjectSample)app.getBean("aopSample");

        obj.getMessage();
        //getMsg!

        obj.getPointCutMessage();
        //before method execution!
        //get point cut Msg!
    }
}

得到以上结果,就证明已经成功的完成了。我们在不修改业务逻辑代码的情况下,成功地在方法执行前加上了自己想要执行的代码。

Spring AOP 实现分析

在引言中提到过,AOP其实是Java代理的一种增强,所以我们也能够想到,最后会生成一个代理的对象。

但是,具体的生成过程又是怎样的呢? 我们下面先通过一个时序图来进行分析。

AOP Proxy对象生成过程

因为最后生成的对象仍旧是一个Bean对象,所以前面仍然是调用getBean方法获取的。真正产生差异的地方在于,我们定义的bean是ProxyFactoryBean,它实现了FactoryBean这个接口。

public interface FactoryBean<T> {
    //返回Bean实例
    T getObject() throws Exception;
    //返回Bean的具体类型
    Class<?> getObjectType();
    //这个bean是否为单例
    boolean isSingleton();
}

因为必须要实现getBoject方法,所以我们就能够自定义生成代理对象的时候需要经历哪些步骤了。

首先,会根据我们在property中声明的interceptorNames来初始化advisors(如果代理对象不是单例,初始化的advisors也不是真正的bean对象,advisors会在newPrototypeInstance时再进行初始化)。

其次,就是根据advisors生成cglib的代理对象。在Spring里面会根据类的定义/bean的定义,决定要生成jdk原生的代理还是cglib的代理。因为我们示例中的类没有实现任何接口,使用的是cglib的代理实现。
CglibAopProxy.getProxy

public Object getProxy(ClassLoader classLoader) {
        //省略debug代码

        try {
            Class<?> rootClass = this.advised.getTargetClass();
            Assert.state(rootClass != null, "Target class must be available for creating a CGLIB proxy");

            Class<?> proxySuperClass = rootClass;
            if (ClassUtils.isCglibProxyClass(rootClass)) {
                proxySuperClass = rootClass.getSuperclass();
                Class<?>[] additionalInterfaces = rootClass.getInterfaces();
                for (Class<?> additionalInterface : additionalInterfaces) {
                    this.advised.addInterface(additionalInterface);
                }
            }

            validateClassIfNecessary(proxySuperClass, classLoader);

            // 对cglib enhancer进行配置
            Enhancer enhancer = createEnhancer();
            if (classLoader != null) {
                enhancer.setClassLoader(classLoader);
                if (classLoader instanceof SmartClassLoader &&
                        ((SmartClassLoader) classLoader).isClassReloadable(proxySuperClass)) {
                    enhancer.setUseCache(false);
                }
            }
            enhancer.setSuperclass(proxySuperClass);

                enhancer.setInterfaces(AopProxyUtils.completeProxiedInterfaces(this.advised));
            enhancer.setNamingPolicy(SpringNamingPolicy.INSTANCE);
            enhancer.setStrategy(new ClassLoaderAwareUndeclaredThrowableStrategy(classLoader));

            //根据advisors来生成callback对象
            Callback[] callbacks = getCallbacks(rootClass);
            Class<?>[] types = new Class<?>[callbacks.length];
            for (int x = 0; x < types.length; x++) {
                types[x] = callbacks[x].getClass();
            }
            // 根据advisors生成callbackFilter,确定哪个方法将会被代理。
            enhancer.setCallbackFilter(new ProxyCallbackFilter(
                    this.advised.getConfigurationOnlyCopy(), this.fixedInterceptorMap, this.fixedInterceptorOffset));
            enhancer.setCallbackTypes(types);

            // 创建代理对象
            return createProxyClassAndInstance(enhancer, callbacks);
        }
        //异常处理
    }

而对应的类图,则如下图所示。
AOP - Cglib Class Diagram
这里看大图

小结

使用ProxyFactoryBean实现的AOP结构较为简单,原理就是通过我们配置的属性(interceptors),生成对应的callback以及callbackfilter,最后通过cglib生成动态代理类,实现代理的功能。

而对于使用JDK原生代理的,也不难想象代理的原理是通过配置的属性,生成需要代理的接口以及设置对应的代理方法。

下一篇文章,我们会对AspectJ中的代理进行分析。

参考文章

  • 《Spring in Action 第三版》
  • 《深入分析Java Web技术内幕》
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值