Spring配置通知的方式+切入点表达式

2024-01-08 03:18

本文主要是介绍Spring配置通知的方式+切入点表达式,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 基于xml的spring-aop配置
    • 切入点表达式
    • 配置通知
    • 环绕通知
  • 基于注解+xml的spring-aop配置
    • 环绕通知
  • 基于纯注解的spring-aop配置
    • 创建配置类
    • 获取容器

基于xml的spring-aop配置

  1. 把通知bean也交给spring管理
  2. 使用aop:config标签表明开始AOP的配置
  3. 使用aop:aspect标签表明开始配置切面
    id:给切面提供唯一标识
    ref:指定通知类bean的id
  4. aop:aspect标签的内部使用对应的标签来配置通知的类型
    让printLog方法在切入点方法执行之前执行,前置通知 aop:before
    method属性:指定Logger类中哪个方法是前置通知。
    pointcut属性:指定切入点表达式,该表达式的含义指的是对业务层中哪些方法增强。

切入点表达式

  • 写法:execution(访问修饰符 返回值 包名.包名……类名.方法名(参数列表))

  • 例:execution(public void com.smday.service.impl.AccountServiceImpl.saveAccount())

  • 访问修饰符可以省略,返回值可以使用通配符*匹配。

  • 包名也可以使用*匹配,数量代表包的层级,当前包可以使用..标识,例如* *..AccountServiceImpl.saveAccount()

  • 类名和方法名也都可以使用*匹配:* *..*.*()

  • 参数列表使用..可以标识有无参数均可,且参数可为任意类型。

全通配写法:* .*(…)

通常情况下,切入点应当设置再业务层实现类下的所有方法:* com.smday.service.impl.*.*(..)

配置通知

	<!--spring-ioc配置--><bean id="accountService" class="com.smday.service.impl.AccountServiceImpl"></bean><!--spring-aop配置--><!--配置Logger类--><bean id="logger" class="com.smday.utils.Logger"></bean><!--配置aop--><aop:config><!--配置切入点表达式 id 指定表达式的唯一标志- expression属性指定表达式内容(此标签写在aop:aspect标签内部只能当前切面使用) 还可以写在aop:aspect,此时所有切面可用--><aop:pointcut id="pt1" expression="execution(* com.smday.service.impl.*.*(..))"></aop:pointcut><aop:aspect id="logAdvise" ref="logger"><!--配置通知的类型并且通知方法和切入点方法的关联--><aop:before method="printLog" pointcut="execution(* com.smday.service.impl.*.*(..))"></aop:before><!--前置通知:在切入点方法执行之前执行--><aop:before method="beforePrintLog" pointcut-ref="pt1" ></aop:before><!--后置通知:在切入点方法正常执行之后执行,他和异常通知永远只能执行一个--><aop:after-returning method="afterReturningPrintLog" pointcut-ref="pt1" ></aop:after-returning><!--异常通知:在切入点方法执行产生异常之后执行--><aop:after-throwing method="afterThrowingPrintLog" pointcut-ref="pt1"></aop:after-throwing><!--最终通知:无论切入点方法是否正常执行它都会在其后面执行--><aop:after method="afterPrintLog" pointcut-ref="pt1"></aop:after><!--环绕通知--><aop:around method="aroundPrintLog" pointcut-ref="pt1"></aop:around></aop:aspect></aop:config>

环绕通知

<!--环绕通知--><aop:around method="aroundPrintLog" pointcut-ref="pt1"></aop:around>
    /*** 环绕通知** 问题:配置环绕通知后,切入点的方法没有执行,而通知方法执行而来.* 分析:通过对比动态代理中的环绕通知代码,发现动态代理的环绕通知有明确的切入点方法调用,而我们代码中没有.* 解决:spring框架提供了ProceedingJoinPoint接口,接口中的proceed()方法,相当于明确调用切入点方法.* 该接口可以作为环绕通知的方法参数,在程序执行时,spring框架我会为我们提供该接口的实现类供我们调用** spring中的环绕通知:*      spring框架为我们提供的一种可以在代码中手动控制增强方法何时执行的方式*/public Object aroundPrintLog(ProceedingJoinPoint pjp){Object rtValue = null;try{//得到方法执行的所需参数Object[] args = pjp.getArgs();System.out.println("前置");//明确调用业务层的方法(切入点方法)rtValue = pjp.proceed(args);System.out.println("后置");}catch (Throwable t){System.out.println("异常");throw new RuntimeException(t);}finally {System.out.println("最终");}return rtValue;}

基于注解+xml的spring-aop配置

<?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:aop="http://www.springframework.org/schema/aop"xmlns:context="http://www.springframework.org/schema/context"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.xsdhttp://www.springframework.org/schema/contexthttp://www.springframework.org/schema/context/spring-context.xsd"><!--配置spring创建容器时要扫描的包--><context:component-scan base-package="com.smday"></context:component-scan><!--配置spring开启注解aop的支持--><aop:aspectj-autoproxy></aop:aspectj-autoproxy></beans>
@Component("logger")
@Aspect
public class Logger {@Pointcut("execution(* com.smday.service.impl.*.*(..))")private void pt1(){}@Before("pt1()")public void beforePrintLog() {System.out.println("Logger.beforePrintLog");}@AfterReturning("pt1()")public void afterReturningPrintLog() {System.out.println("Logger.afterReturningPrintLog");}@AfterThrowing("pt1()")public void afterThrowingPrintLog() {System.out.println("Logger.afterThrowingPrintLog");}@After("pt1()")public void afterPrintLog() {System.out.println("Logger.afterPrintLog");}
}

基于注解的方式,最终通知的调用将会出现在后置或者异常通知前,建议使用环绕通知。

环绕通知

@Component("logger")
@Aspect
public class Logger {@Pointcut("execution(* com.smday.service.impl.*.*(..))")private void pt1(){}@Around("pt1()")public Object aroundPrintLog(ProceedingJoinPoint pjp) {Object rtValue = null;try {//得到方法执行的所需参数Object[] args = pjp.getArgs();System.out.println("前置");//明确调用业务层的方法(切入点方法)rtValue = pjp.proceed(args);System.out.println("后置");} catch (Throwable t) {System.out.println("异常");throw new RuntimeException(t);} finally {System.out.println("最终");}return rtValue;}
}

基于纯注解的spring-aop配置

创建配置类

@Configuration
@ComponentScan(basePackages = "com.smday")
@EnableAspectJAutoProxy
public class SpringConfiguration {}

获取容器

public class aopTest {public static void main(String[] args) {//获取容器ApplicationContext ac = new AnnotationConfigApplicationContext(SpringConfiguration.class);//获取对象AccountService as = ac.getBean("accountService", AccountService.class);//执行方法as.saveAccount();}
}

这篇关于Spring配置通知的方式+切入点表达式的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

java如何解压zip压缩包

《java如何解压zip压缩包》:本文主要介绍java如何解压zip压缩包问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java解压zip压缩包实例代码结果如下总结java解压zip压缩包坐在旁边的小伙伴问我怎么用 java 将服务器上的压缩文件解压出来,

SpringBoot中SM2公钥加密、私钥解密的实现示例详解

《SpringBoot中SM2公钥加密、私钥解密的实现示例详解》本文介绍了如何在SpringBoot项目中实现SM2公钥加密和私钥解密的功能,通过使用Hutool库和BouncyCastle依赖,简化... 目录一、前言1、加密信息(示例)2、加密结果(示例)二、实现代码1、yml文件配置2、创建SM2工具

Spring WebFlux 与 WebClient 使用指南及最佳实践

《SpringWebFlux与WebClient使用指南及最佳实践》WebClient是SpringWebFlux模块提供的非阻塞、响应式HTTP客户端,基于ProjectReactor实现,... 目录Spring WebFlux 与 WebClient 使用指南1. WebClient 概述2. 核心依

SQL Server配置管理器无法打开的四种解决方法

《SQLServer配置管理器无法打开的四种解决方法》本文总结了SQLServer配置管理器无法打开的四种解决方法,文中通过图文示例介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录方法一:桌面图标进入方法二:运行窗口进入检查版本号对照表php方法三:查找文件路径方法四:检查 S

Spring Boot @RestControllerAdvice全局异常处理最佳实践

《SpringBoot@RestControllerAdvice全局异常处理最佳实践》本文详解SpringBoot中通过@RestControllerAdvice实现全局异常处理,强调代码复用、统... 目录前言一、为什么要使用全局异常处理?二、核心注解解析1. @RestControllerAdvice2

Spring IoC 容器的使用详解(最新整理)

《SpringIoC容器的使用详解(最新整理)》文章介绍了Spring框架中的应用分层思想与IoC容器原理,通过分层解耦业务逻辑、数据访问等模块,IoC容器利用@Component注解管理Bean... 目录1. 应用分层2. IoC 的介绍3. IoC 容器的使用3.1. bean 的存储3.2. 方法注

Spring事务传播机制最佳实践

《Spring事务传播机制最佳实践》Spring的事务传播机制为我们提供了优雅的解决方案,本文将带您深入理解这一机制,掌握不同场景下的最佳实践,感兴趣的朋友一起看看吧... 目录1. 什么是事务传播行为2. Spring支持的七种事务传播行为2.1 REQUIRED(默认)2.2 SUPPORTS2

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

Java进程异常故障定位及排查过程

《Java进程异常故障定位及排查过程》:本文主要介绍Java进程异常故障定位及排查过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、故障发现与初步判断1. 监控系统告警2. 日志初步分析二、核心排查工具与步骤1. 进程状态检查2. CPU 飙升问题3. 内存

java中新生代和老生代的关系说明

《java中新生代和老生代的关系说明》:本文主要介绍java中新生代和老生代的关系说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、内存区域划分新生代老年代二、对象生命周期与晋升流程三、新生代与老年代的协作机制1. 跨代引用处理2. 动态年龄判定3. 空间分