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

举报
小工匠 发表于 2021/09/10 23:07:20 2021/09/10
【摘要】 概述在Spring中配置 通过配置文件实现增强 配置文件ProxyFactory Bean配置文件当中常用的属性测试类 其他代码 概述 在上篇博文的案...

概述

在上篇博文的案例中

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

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

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

Java-CGLib动态代理

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

我们看下的源码 ProxyFactoryBean#setProxyInterfaces

这里写图片描述

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

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

这里写图片描述

  • 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);
  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8

使用CGLib代理实现则为:

ProxyFactory factory=new ProxyFactory();
//设置代理接口
factory.setInterfaces(target.getClass().getInterfaces());
//启用优化,将启用cglib代理方式
factory.setOptimize(true);
//设置代理类
factory.setTarget(target);
//添加增强类
factory.addAdvice(advice);
  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

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


在Spring中配置

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

通过配置文件实现增强

这里写图片描述

配置文件

<?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>

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17

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");
    }
}

  
 
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19

运行单元测试:
这里写图片描述

其他代码

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

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

文章来源: artisan.blog.csdn.net,作者:小小工匠,版权归原作者所有,如需转载,请联系作者。

原文链接:artisan.blog.csdn.net/article/details/77175411

【版权声明】本文为华为云社区用户转载文章,如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。