Spring学习(8)-AOP之ProxyFactoryBean、RegexMethodPointcutAdvisor、BeanNameAutoProxyCreator

本文主要是介绍Spring学习(8)-AOP之ProxyFactoryBean、RegexMethodPointcutAdvisor、BeanNameAutoProxyCreator,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言:

上一篇学习了Spring AOP(面向切面编程)的基础,在Spring Bean 的配置文件中代理工厂ProxyFactoryBean 的写法似乎并没有见到切入点(pointcut),切面(aspect),织入(weaving)等等的英文所在,那是从哪里来的这些概念呢?这篇文章就告诉你aop的真正写法。

1 配置ProxyFactoryBean

上篇博客地址:Spring学习(7)-AOP面向切面编程

前言:

上一篇对于spring aop仅仅使用代理工厂ProxyFactoryBean ,这是最基础的spring aop写法,而且每一个通知/增强类都不是POJO(简单java对象),它都是实现了Spring提供的接口,复习一下:

  • 前置通知:MethodBeforeAdvice
  • 后置通知:AfterReturningAdvice
  • 环绕通知:MethodInterceptor(只有这个是aopalliance包内的,其他均为spring aop包)
  • 异常通知:ThrowsAdvice
  • 引介通知:IntroductionInterceptor

然后我们在xml文件中配置一个ProxyFactoryBean的 Bean,将实现接口的通知类放到代理工厂的interceptorNames属性中,再指定代理工厂代理的主业务接口实现主业务接口的目标类,将这些通知织入到目标类中,实现aop的思想。

我就再使用一下上次的例子:

BuyPhoneAroundAdvice (买手机的环绕通知类)

package com.cheng.spring.advice;import org.aopalliance.intercept.MethodInterceptor;
import org.aopalliance.intercept.MethodInvocation;//环绕通知需要实现MethodInterceptor接口
//环绕通知:在主业务的前,后执行服务功能
public class BuyPhoneAroundAdvice implements MethodInterceptor {@Overridepublic Object invoke(MethodInvocation invocation) throws Throwable {//环绕通知的前置System.out.println("同学,您定制的钛金手机给您送来了,赶紧拍照发朋友圈---环绕前置");//执行核心业务,作为环绕的前后通知分割点invocation.proceed();System.out.println("用了两天就坏了,智商税还交的不够啊!---环绕后置");//环绕通知的后置return null;}
}

BuyPhone(买手机的主业务接口)

package com.cheng.spring.service;//主业务,购买手机服务
public interface BuyPhoneService {//编写业务方法public void giveMeAphone(String name, String pass);
}

BuyPhoneImpl (买手机的主业务实现类)

ackage com.cheng.spring.serviceImpl;
import com.cheng.spring.service.BuyPhoneService;public class BuyPhoneServiceImpl implements BuyPhoneService{@Overridepublic void giveMeAphone(String name, String pass) {System.out.println("我的身份只有8848钛金手机才配得上,8848你值得拥有");}
}

我想通过aop来将环绕通知类切入到我的买手机主业务实现类,需要在applicationContext.xml中配置这些:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:p="http://www.springframework.org/schema/p"xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsd"><!-- 配置目标对象--><bean id="buyPhoneService" class="com.cheng.spring.serviceImpl.BuyPhoneServiceImpl"/><!-- 配置切面:环绕通知 --><bean id="buyPhoneBeforeAdvice" class="com.cheng.spring.advice.BuyPhoneBeforeAdvice"/><!-- 使用代理工厂将切面织入目标对象 --><bean id="factoryBean" class="org.springframework.aop.framework.ProxyFactoryBean"p:proxyInterfaces="com.cheng.spring.service.BuyPhoneService"p:target-ref="buyPhoneService"p:interceptorNames="buyPhoneAroundAdvice"p:proxyTargetClass="false"/>
</beans>

这样我们通过容器拿到代理工厂返回的接口,来调用它的方法

package com.cheng.spring.test;import com.cheng.spring.service.BuyPhone;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;public class Test {public static void main(String[] args) {//如何通过context模块加载配置文件,获取bean实例ApplicationContext context = newClassPathXmlApplicationContext("applicationContext.xml");//如果要启动主业务,我需要获取哪个bean的实例?BuyPhone buyPhone = (BuyPhone)context.getBean("factoryBean");buyPhone.giveMeAphone("此成", "123");}
}

结果:
在这里插入图片描述

后语:

只配置代理工厂会对目标类的所有方法拦截

上方实现一个简单环绕通知,且是对买手机这个实现类的所有方法起作用的,什么意思呢?
我给接口和实现类加一个方法:

package com.cheng.spring.serviceImpl;
import com.cheng.spring.service.BuyPhoneService;public class BuyPhoneServiceImpl implements BuyPhoneService{@Overridepublic void giveMeAphone(String name, String pass) {System.out.println("我的身份只有8848钛金手机才配得上,8848你值得拥有");}@Overridepublic void newMethod() {System.out.println("甩卖8848,只要998");}
}

我把新加的方法也运行一下

public class Test {public static void main(String[] args) {//通过context模块加载配置文件,获取bean实例ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");//获得代理工厂返回的接口BuyPhone buyPhone = (BuyPhone)context.getBean("factoryBean");buyPhone.giveMeAphone("此成", "123");buyPhone.newMethod();}
}

结果:
在这里插入图片描述
我本来只想要giveMeAphone方法用这个环绕通知的,但这个实现类的所有方法都加上这个通知了,这并不是我想要的。

2 配置RegexMethodPointcutAdvisor

在spring aop 的support里有这个帮助类,帮助我们实现单个或多个指定方法的拦截(织入)

它相当于是一个过渡类,把目标对象的id先放到这个RegexMethodPointcutAdvisor的active属性内,再给patterns赋值告诉它你只需要什么方法被织入通知。最后把这个Bean的id再给原来的代理工厂,即可实现指定方法的切入。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:p="http://www.springframework.org/schema/p"xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsd"><!-- 配置目标对象--><bean id="buyPhoneService" class="com.cheng.spring.serviceImpl.BuyPhoneServiceImpl"/><!-- 配置切面:环绕通知 --><bean id="buyPhoneBeforeAdvice" class="com.cheng.spring.advice.BuyPhoneBeforeAdvice"/><!-- 使用RegexpMethodPointcutAdvisor指定切入的是哪个方法 --><!--patterns,如果有多个指定的值的话,可以使用,隔开,例如value=".*giveMeAphone,.*newMethod"--><bean id="advisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"p:advice-ref="buyPhoneAroundAdvice"p:patterns=".*giveMeAphone"/><!-- 使用代理工厂将切面织入目标对象 --><bean id="factoryBean" class="org.springframework.aop.framework.ProxyFactoryBean"p:proxyInterfaces="com.cheng.spring.service.BuyPhoneService"p:target-ref="buyPhoneService"p:interceptorNames="buyPhoneAroundAdvice"p:proxyTargetClass="false"/>
</beans>

再次运行一下:
在这里插入图片描述
完成我想要的单个方法的织入。

3 配置BeanNameAutoProxyCreator

它是帮助我们减少配置一堆代理工厂的类

比如说我上面配置的一个代理工厂的Bean,它只能帮助我把一个/多个切面织入到另一个目标对象中,如果我想多个目标对象都用这个切面,我就要配置多个代理工厂,这就造成了代码的重复。而且我要获得生产好的代理返回的Bean,还需要去配置里找代理工厂的id,实在有些麻烦。

使用BeanNameAutoProxyCreator可以方便的获得代理后的目标对象接口
这个类在Spring aop包中的framework中的autoproxy中

BeanNameAutoProxyCreator的属性:

  • p:beanNames=“你需要切入的主业务bean id” (可以写*Service,将自动找到以Service结尾的bean id)
  • p:interceptorNames=“切面id”

使用BeanNameAutoProxyCreator后,我们的配置变成这样

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:p="http://www.springframework.org/schema/p"xmlns:aop="http://www.springframework.org/schema/aop"xsi:schemaLocation="http://www.springframework.org/schema/beanshttp://www.springframework.org/schema/beans/spring-beans.xsdhttp://www.springframework.org/schema/aophttp://www.springframework.org/schema/aop/spring-aop.xsd"><!-- 配置切入点--><bean id="buyPhoneService" class="com.cheng.spring.serviceImpl.BuyPhoneServiceImpl"/><!-- 配置切入面:环绕通知 --><bean id="buyPhoneAroundAdvice" class="com.cheng.spring.advice.BuyPhoneAroundAdvice"/><!-- 使用RegexpMethodPointcutAdvisor指定切入的是哪个方法 --><!--patterns,如果有多个指定的值的话,可以使用,隔开,例如value=".*giveMeAphone,.*newMethod"--><bean id="advisor" class="org.springframework.aop.support.RegexpMethodPointcutAdvisor"p:advice-ref="buyPhoneAroundAdvice"p:patterns=".*giveMeAphone"/><!-- 自动返回代理工厂 --><bean class="org.springframework.aop.framework.autoproxy.BeanNameAutoProxyCreator"p:beanNames="*Service"p:interceptorNames="advisor"/>
</beans>

还是运行之前的主函数:
在这里插入图片描述
可以看到结果是一致的。
通过BeanNameAutoProxyCreator,我们可以配置多个目标对象来织入我们需要的切面。

4 总结

  1. ProxyFactoryBean :代理工厂,最经典基础的aop的配置,只使用它的话,会让切面(通知)织入到目标对象的所有方法内;
  2. RegexMethodPointcutAdvisor:方法切入,配合ProxyFactoryBean 使用,可以只让切面(通知)织入到目标对象的某几个方法内;
  3. BeanNameAutoProxyCreator:自动创建代理对象,如果需要一个切面织入到多个目标对象中,它是很棒的选择。

下一篇:Spring学习(9)-AOP之使用aop:config标签

文毕,如有助,赞之吾嗨~

这篇关于Spring学习(8)-AOP之ProxyFactoryBean、RegexMethodPointcutAdvisor、BeanNameAutoProxyCreator的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/184764

相关文章

Java利用docx4j+Freemarker生成word文档

《Java利用docx4j+Freemarker生成word文档》这篇文章主要为大家详细介绍了Java如何利用docx4j+Freemarker生成word文档,文中的示例代码讲解详细,感兴趣的小伙伴... 目录技术方案maven依赖创建模板文件实现代码技术方案Java 1.8 + docx4j + Fr

SpringBoot首笔交易慢问题排查与优化方案

《SpringBoot首笔交易慢问题排查与优化方案》在我们的微服务项目中,遇到这样的问题:应用启动后,第一笔交易响应耗时高达4、5秒,而后续请求均能在毫秒级完成,这不仅触发监控告警,也极大影响了用户体... 目录问题背景排查步骤1. 日志分析2. 性能工具定位优化方案:提前预热各种资源1. Flowable

基于SpringBoot+Mybatis实现Mysql分表

《基于SpringBoot+Mybatis实现Mysql分表》这篇文章主要为大家详细介绍了基于SpringBoot+Mybatis实现Mysql分表的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录基本思路定义注解创建ThreadLocal创建拦截器业务处理基本思路1.根据创建时间字段按年进

Java编译生成多个.class文件的原理和作用

《Java编译生成多个.class文件的原理和作用》作为一名经验丰富的开发者,在Java项目中执行编译后,可能会发现一个.java源文件有时会产生多个.class文件,从技术实现层面详细剖析这一现象... 目录一、内部类机制与.class文件生成成员内部类(常规内部类)局部内部类(方法内部类)匿名内部类二、

SpringBoot实现数据库读写分离的3种方法小结

《SpringBoot实现数据库读写分离的3种方法小结》为了提高系统的读写性能和可用性,读写分离是一种经典的数据库架构模式,在SpringBoot应用中,有多种方式可以实现数据库读写分离,本文将介绍三... 目录一、数据库读写分离概述二、方案一:基于AbstractRoutingDataSource实现动态

Springboot @Autowired和@Resource的区别解析

《Springboot@Autowired和@Resource的区别解析》@Resource是JDK提供的注解,只是Spring在实现上提供了这个注解的功能支持,本文给大家介绍Springboot@... 目录【一】定义【1】@Autowired【2】@Resource【二】区别【1】包含的属性不同【2】@

springboot循环依赖问题案例代码及解决办法

《springboot循环依赖问题案例代码及解决办法》在SpringBoot中,如果两个或多个Bean之间存在循环依赖(即BeanA依赖BeanB,而BeanB又依赖BeanA),会导致Spring的... 目录1. 什么是循环依赖?2. 循环依赖的场景案例3. 解决循环依赖的常见方法方法 1:使用 @La

Java枚举类实现Key-Value映射的多种实现方式

《Java枚举类实现Key-Value映射的多种实现方式》在Java开发中,枚举(Enum)是一种特殊的类,本文将详细介绍Java枚举类实现key-value映射的多种方式,有需要的小伙伴可以根据需要... 目录前言一、基础实现方式1.1 为枚举添加属性和构造方法二、http://www.cppcns.co

Elasticsearch 在 Java 中的使用教程

《Elasticsearch在Java中的使用教程》Elasticsearch是一个分布式搜索和分析引擎,基于ApacheLucene构建,能够实现实时数据的存储、搜索、和分析,它广泛应用于全文... 目录1. Elasticsearch 简介2. 环境准备2.1 安装 Elasticsearch2.2 J

Java中的String.valueOf()和toString()方法区别小结

《Java中的String.valueOf()和toString()方法区别小结》字符串操作是开发者日常编程任务中不可或缺的一部分,转换为字符串是一种常见需求,其中最常见的就是String.value... 目录String.valueOf()方法方法定义方法实现使用示例使用场景toString()方法方法