概述

在上篇博文的案例中

Spring-AOP 创建增强类-前置增强

我们使用org.springframework.aop.framework.ProxyFactory通过编码的方式将GreetingBeforeAdvice的增强织入目标类NaiveWaiter中

结合我们前面说的
Java-JDK动态代理

Java-CGLib动态代理

动态代理技术是不是很相似?

我们看下的源码 ProxyFactoryBean#setProxyInterfaces

Spring-AOP 通过配置文件实现 前置增强_ProxyFacto

可以发现,ProxyFactoryBean的内部其实就是使用JDK或者CGLib动态代理技术将增强应用到目标类中的。

Spring定义的AOPProxy类具有两个final类型的实现类,如下图所示

Spring-AOP 通过配置文件实现 前置增强_java_02

  • Cglib2AopProxy是使用cglib代理技术来创建代理

  • JdkDynamicAopProxy是使用jdk代理技术来创建代理

如果通过ProxyFactory的setInterfaces(Class[] interfaces)方法指定目标接口进行代理,则ProxyFactory使用JdkDynamaicAopProxy,如果针对类的代理,则使用Cglib2AopProxy.

此外,还可以通过ProxyFactory的setOptimize(true)方法让ProxyFactory启动优化代理方式,这样,针对接口的代理也会使用Cglib2AopProxy。

我们使用JDK代理来实现 Spring-AOP 增强(Advice)类型和创建增强类 中的业务测试类

//创建代理工厂对象
ProxyFactory factory=new ProxyFactory();
//设置代理接口
factory.setInterfaces(target.getClass().getInterfaces());
//设置代理类
factory.setTarget(target);
//设置增强类
factory.addAdvice(advice);

使用CGLib代理实现则为:

ProxyFactory factory=new ProxyFactory();
//设置代理接口
factory.setInterfaces(target.getClass().getInterfaces());
//启用优化,将启用cglib代理方式
factory.setOptimize(true);
//设置代理类
factory.setTarget(target);
//添加增强类
factory.addAdvice(advice);

ProxyFactory通过addAdvice来增加一个增强。 用户可以使用该方法增加多个增强,通过增强形成一个增强链,他们的调用顺序和添加顺序是一致的


在Spring中配置

代码已托管到Github—> https://github.com/yangshangwei/SpringMaster

通过配置文件实现增强

Spring-AOP 通过配置文件实现 前置增强_ProxyFacto_03

配置文件

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
    xmlns:p="http://www.springframework.org/schema/p"
    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
    xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean id="target" class="com.xgj.aop.spring.advice.proxyFactory.NaiveWaiter"/>

    <bean id="greetingBeforeAdvice" class="com.xgj.aop.spring.advice.proxyFactory.GreetingBeforeAdvice"/>

    <bean id="waiter" class="org.springframework.aop.framework.ProxyFactoryBean"
        p:proxyInterfaces="com.xgj.aop.spring.advice.proxyFactory.Waiter"
        p:interceptorNames="greetingBeforeAdvice"
        p:target-ref="target"/>

</beans>

ProxyFactory Bean配置文件当中常用的属性

  • target:我们需要代理的目标对象

  • proxyInterfaces:代理所要实现的接口,可以是多个接口

  • interceptorNames:需要织入的目标对象的Bean的列表(增强类的Bean列表),使用Bean的名称来指定。

  • singleton:确定返回的代理是不是单实例的,系统默认返回的是单实例的。

  • optimize:当值为true时,强制使用cglib代理。当是singleton的实例时我们推荐使用cglib代理,当是其他作用域的时候,推荐使用JDK的代理。原因是cglib创建代理速度比较慢,但是运行效率高。JDK代理则刚好相反。

  • proxyTargetClass:是否对类进行代理而不是对接口进行代理,当值为true的时候使用cglib代理

将proxyTargetClass设置为true后,无需再设置proxyInterfaces属性,即使设置了也会被ProxyFactory忽略

测试类

package com.xgj.aop.spring.advice.proxyFactory;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class BeforceAdivceByXmlTest {

    @Test
    public void test() {
        ApplicationContext ctx = new ClassPathXmlApplicationContext(
                "classpath:com/xgj/aop/spring/advice/proxyFactory/beans.xml");

        Waiter waiter = ctx.getBean("waiter", Waiter.class);
        waiter.greetTo("Xiao");
        waiter.serverTo("GongJiang");
    }
}

运行单元测试:
Spring-AOP 通过配置文件实现 前置增强_java_04

其他代码

业务接口类:Waiter.Java
业务实现类:NativeWaiter.java
业务增强类:GreetingBeforeAdvice.java

参考 Spring-AOP 增强(Advice)类型和创建增强类#实例-通过代码实现增强