Spring详解(七)----AspectJ 实现AOP

2024-08-29 13:08

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

原文网址:https://www.cnblogs.com/ysocean/p/7507993.html

AspectJ 实现AOP
目录
1、什么是 AspectJ?
2、切入点表达式
2、Aspect 通知类型
3、AOP具体实例
4、测试异常通知
5、测试环绕通知


上一篇博客我们引出了 AOP 的概念,以及 AOP 的具体实现方式。但是为什么要这样实现?以及提出的切入点表达式到底该怎么理解?
  这篇博客我们通过对 AspectJ 框架的介绍来详细了解。

1、什么是 AspectJ?
  AspectJ是一个面向切面的框架,它扩展了Java语言。AspectJ定义了AOP语法,也可以说 AspectJ 是一个基于 Java 语言的 AOP 框架。通常我们在使用 Spring AOP 的时候,都会导入 AspectJ 的相关 jar 包。
在这里插入图片描述

在 spring2.0以后,spring新增了对AspectJ 切点表达式的支持;Aspect1.5新增注解功能,通过 JDK5的注解技术,能直接在类中定义切面;新版本的 spring 框架,也都建议使用 AspectJ 来实现 AOP。所以说在 spring AOP 的核心包 Spring-aop-3.2.jar 里面也有对 AspectJ 的支持。

2、切入点表达式
  上一篇博客中,我们在spring配置文件中配置如下:

<aop:pointcut expression=“execution(* com.ys.aop..(…))” id=“myPointCut”/>
  那么它表达的意思是 返回值任意,包名为 com.ys.aop 下的任意类名中的任意方法名,参数任意。那么这到底是什么意思呢?
  首先 execution 是 AspectJ 框架定义的一个切入点函数,其语法形式如下:
execution(modifiers-pattern? ref-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)类修饰符 返回值 方法所在的包 方法名 方法抛出的异常
  简单点来说就是:
1 语法:execution(修饰符 返回值 包.类.方法名(参数) throws异常)
  具体解释我们用下面一张思维导图来看:
在这里插入图片描述

注意:如果切入点表达式有多个不同目录呢? 可以通过 || 来表示或的关系。  
<aop:pointcut expression=“execution(* com.ys.Service1.(…)) || execution(* com.ys.Service2.(…))” id=“myPointCut”/>
  表示匹配 com.ys包下的,以 Service1结尾或者以Service2结尾的类的任意方法。
  
  AOP 切入点表达式支持多种形式的定义规则:
1、execution:匹配方法的执行(常用)
execution(public .(…))
2.within:匹配包或子包中的方法(了解)
within(com.ys.aop…*)
3.this:匹配实现接口的代理对象中的方法(了解)
this(com.ys.aop.user.UserDAO)
4.target:匹配实现接口的目标对象中的方法(了解)
target(com.ys.aop.user.UserDAO)
5.args:匹配参数格式符合标准的方法(了解)
args(int,int)6.bean(id)
6对指定的bean所有的方法(了解)
bean(‘userServiceId’)

2、Aspect 通知类型
Aspect 通知类型,定义了类型名称以及方法格式。类型如下:
before:前置通知(应用:各种校验)
在方法执行前执行,如果通知抛出异常,阻止方法运行afterReturning:
后置通知(应用:常规数据处理)
方法正常返回后执行,如果方法中抛出异常,通知无法执行 必须在方法执行后才执行,所以可以获得方法的返回值。
around:环绕通知(应用:十分强大,可以做任何事情) 方法执行前后分别执行,可以阻止方法的执行 必须手动执行目标方法
afterThrowing:抛出异常通知(应用:包装异常信息) 方法抛出异常后执行,如果方法没有抛出异常,无法执行
after:最终通知(应用:清理现场) 方法执行完毕后执行,无论方法中是否出现异常
  这里最重要的是around,环绕通知,它可以代替上面的任意通知。
  在程序中表示的意思如下:
123456789 try{ //前置:before //手动执行目标方法 //后置:afterRetruning} catch(){ //抛出异常 afterThrowing} finally{ //最终 after}
  对应的 jar 包如下:
  在这里插入图片描述

我们可以查看源码:
  
在这里插入图片描述
在这里插入图片描述
3、AOP具体实例
  ①、创建接口
package com.ys.aop; public interface UserService {
//添加 user
public void addUser();
//删除 user
public void deleteUser();}
  ②、创建实现类
package com.ys.aop; public class UserServiceImpl implements UserService{
@Override
public void addUser() {
System.out.println(“增加 User”);
}
@Override
public void deleteUser() {
System.out.println(“删除 User”);
}
}
  ③、创建切面类(包含各种通知)  
package com.ys.aop; import org.aspectj.lang.JoinPoint; public class MyAspect {
/** * JoinPoint 能获取目标方法的一些基本信息

  • @param joinPoint
  • */
  • public void myBefore(JoinPoint joinPoint){
  • 	System.out.println("前置通知 : " + joinPoint.getSignature().getName());    
    
  • }
  • public void myAfterReturning(JoinPoint joinPoint,Object ret){
  • 	System.out.println("后置通知 : " + joinPoint.getSignature().getName() + " , -->" + ret);    
    
  • }
  • public void myAfter(){ System.out.println(“最终通知”);
  • }
    }
      ④、创建spring配置文件applicationContext.xml
      我们首先测试前置通知、后置通知、最终通知

aop:config <aop:aspect ref=“myAspect”>

<aop:pointcut expression=“execution(* com.ys.aop..(…))” id=“myPointCut”/>

<aop:before method="myBefore" pointcut-ref="myPointCut"/>                          
<!-- 3.2后置通知  ,目标方法后执行,获得返回值                
<aop:after-returning method="" pointcut-ref="" returning=""/>                   returning 通知方法第二个参数的名称                通知方法格式:public void myAfterReturning(JoinPoint joinPoint,Object ret){                    参数1:连接点描述                    参数2:类型Object,参数名 returning="ret" 配置的        -->        <aop:after-returning method="myAfterReturning" pointcut-ref="myPointCut" returning="ret" />                     <!-- 3.3 最终通知 -->                <aop:after method="myAfter" pointcut-ref="myPointCut"/>                       </aop:aspect>    </aop:config>
  ⑤、测试 @Test public void testAop(){ ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml"); UserService useService = (UserService) context.getBean("userService"); useService.addUser(); }   控制台打印:   ![在这里插入图片描述](https://img-blog.csdnimg.cn/20190802082245821.?x-oss-process=image/watermark,type_ZmFuZ3poZW5naGVpdGk,shadow_10,text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3UwMTMxMzE3MTY=,size_16,color_FFFFFF,t_70)

注意,后置通知的返回值为 null,是因为我们的目标方法 addUser() 没有返回值。如果有返回值,这里就是addUser() 的返回值。

4、测试异常通知
  目标接口保持不变,目标类我们手动引入异常:
public void addUser() {
int i = 1/0;
//显然这里会抛出除数不能为 0
System.out.println(“增加 User”);
}
  接着配置切面:MyAspect.java
public void myAfterThrowing(JoinPoint joinPoint,Throwable e){
System.out.println("抛出异常通知 : " + e.getMessage());
}
  接着在 applicationContext.xml 中配置如下:

<aop:after-throwing method=“myAfterThrowing” pointcut-ref=“myPointCut” throwing=“e”/>
  测试:
@Test
public void testAop(){
String str = “com/ys/execption/applicationContext.xml”;
ApplicationContext context = new ClassPathXmlApplicationContext(str);
UserService useService = (UserService) context.getBean(“userService”); useService.addUser();
}
  控制台打印:
  在这里插入图片描述

5、测试环绕通知
  目标接口和目标类保持不变,切面MyAspect 修改如下:
public class MyAspect {
public Object myAround(ProceedingJoinPoint joinPoint) throws Throwable{ System.out.println(“前置通知”);
//手动执行目标方法
Object obj = joinPoint.proceed();
System.out.println(“后置通知”);
return obj;
}
}
  applicationContext.xml 配置如下:

<aop:around method=“myAround” pointcut-ref=“myPointCut”/>
  测试:
@Test
public void testAop(){
String str = “com/ys/around/applicationContext.xml”;
ApplicationContext context = new ClassPathXmlApplicationContext(str);
UserService useService = (UserService) context.getBean(“userService”); useService.addUser();
}
  打印结果:
  在这里插入图片描述

那么至此,通过 xml 配置的方式我们讲解了Spring AOP 的配置。下一章将通过注解的方式来实现。

这篇关于Spring详解(七)----AspectJ 实现AOP的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Conda与Python venv虚拟环境的区别与使用方法详解

《Conda与Pythonvenv虚拟环境的区别与使用方法详解》随着Python社区的成长,虚拟环境的概念和技术也在不断发展,:本文主要介绍Conda与Pythonvenv虚拟环境的区别与使用... 目录前言一、Conda 与 python venv 的核心区别1. Conda 的特点2. Python v

Spring Boot中WebSocket常用使用方法详解

《SpringBoot中WebSocket常用使用方法详解》本文从WebSocket的基础概念出发,详细介绍了SpringBoot集成WebSocket的步骤,并重点讲解了常用的使用方法,包括简单消... 目录一、WebSocket基础概念1.1 什么是WebSocket1.2 WebSocket与HTTP

SpringBoot+Docker+Graylog 如何让错误自动报警

《SpringBoot+Docker+Graylog如何让错误自动报警》SpringBoot默认使用SLF4J与Logback,支持多日志级别和配置方式,可输出到控制台、文件及远程服务器,集成ELK... 目录01 Spring Boot 默认日志框架解析02 Spring Boot 日志级别详解03 Sp

Python使用python-can实现合并BLF文件

《Python使用python-can实现合并BLF文件》python-can库是Python生态中专注于CAN总线通信与数据处理的强大工具,本文将使用python-can为BLF文件合并提供高效灵活... 目录一、python-can 库:CAN 数据处理的利器二、BLF 文件合并核心代码解析1. 基础合

java中反射Reflection的4个作用详解

《java中反射Reflection的4个作用详解》反射Reflection是Java等编程语言中的一个重要特性,它允许程序在运行时进行自我检查和对内部成员(如字段、方法、类等)的操作,本文将详细介绍... 目录作用1、在运行时判断任意一个对象所属的类作用2、在运行时构造任意一个类的对象作用3、在运行时判断

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

golang版本升级如何实现

《golang版本升级如何实现》:本文主要介绍golang版本升级如何实现问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录golanwww.chinasem.cng版本升级linux上golang版本升级删除golang旧版本安装golang最新版本总结gola

java如何解压zip压缩包

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

MySQL 中的 CAST 函数详解及常见用法

《MySQL中的CAST函数详解及常见用法》CAST函数是MySQL中用于数据类型转换的重要函数,它允许你将一个值从一种数据类型转换为另一种数据类型,本文给大家介绍MySQL中的CAST... 目录mysql 中的 CAST 函数详解一、基本语法二、支持的数据类型三、常见用法示例1. 字符串转数字2. 数字

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

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