Spring AOP基于注解方式实现和细节

news2024/12/23 14:56:11

目录

一、Spring AOP底层技术

二、初步实现AOP编程

三、获取切点详细信息

四、 切点表达式语法

五、重用(提取)切点表达式


一、Spring AOP底层技术

SpringAop的核心在于动态代理,那么在SpringAop的底层的技术是依靠了什么技术呢?

  • 动态代理(InvocationHandler):JDK原生的实现方式,需要被代理的目标类必须实现接口。因为这个技术要求代理对象和目标对象实现同样的接口(兄弟两个拜把子模式)。
  • cglib:通过继承被代理的目标类实现代理,所以不需要目标类实现接口。
  • AspectJ:早期的AOP实现的框架,SpringAOP借用了AspectJ中的AOP注解。

二、初步实现AOP编程

2.1实现AOP需要以下注解:

注解说明
@SpringJUnitConfig在JUnit测试类中使用Spring测试上下文配置
@Aspect将类标记为切面类,定义切面逻辑和增强方法的位置
@EnableAspectJAutoProxy开启AspectJ自动代理,用于启用Spring AOP的功能

2.2需要导入以下依赖

<!-- 切面实现 -->
 <dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-aspects</artifactId>
    <version>6.0.6</version>
</dependency>
<!-- spring核心 -->
 <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-context</artifactId>
      <version>5.0.2.RELEASE</version>
    </dependency>
<!-- spring-test容器测试 -->
 <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-test</artifactId>
      <version>6.0.6</version>
      <scope>test</scope>
    </dependency>

2.3 增强(通知)注解

注解说明
@Before在目标方法执行前执行的增强逻辑
@AfterReturning在目标方法成功返回后执行的增强逻辑
@AfterThrowing在目标方法抛出异常后执行的增强逻辑
@After在目标方法执行后执行的增强逻辑
@Around包裹目标方法,在目标方法执行前后都可以执行自定义的增强逻辑

实现增强(通知)的步骤

  1. 定义方法存储增强代码
  2. 使用注解配置,指定插入目标的位置
  3. 配置切点表达式(选中插入的方法,切点)
  4. 补全注解,加入到ioc容器,并且设置切面@Aspect
  5. 开启Aspect注解注释

案例代码:

//4.补全注解
@Component
@Aspect

//1.创建增强类与增强方法start(),after,Error
public class advice {

//    2.使用注解配置,配置插入位置@Before @After @AfterThrowing
//    3.配入切点表达式execution(* com.alphamilk.Impl.*.*(..))表明需要插入的方法为所有com.alhpamilk.Impl包下所有类的所有方法
    @Before("execution(* com.alphamilk.Impl.*.*(..))")
    public void  start(){
        System.out.println("方法起始处插入");
    }

    @After("execution(* com.alphamilk.Impl.*.*(..))")
    public void after(){
        System.out.println("方法结束后插入");
    }
    @AfterThrowing("execution(* com.alphamilk.Impl.*.*(..))")
    public void Error(){
        System.out.println("方法异常时候插入");
    }
}
@ComponentScan(value = "com.alphamilk")
@Configuration
//6.注解类中开启注解注释
@EnableAspectJAutoProxy
public class JavaConfig {
}


三、获取切点详细信息

虽然已经初步实现了AOP的实现,但是还不够,在调用多个方法时候如果都是输入,调用方法前,调用方法后等等,这样并不能区分是调用了哪个方法,所以为了区分我们需要获取调用这个方法的相关信息,比如参数,方法名,返回值等等。

具体实现方式:
通过JoinPoint接口的下面几个方法获取

方法说明
getTarget()获取被代理的目标对象
getClass()获取被代理的目标对象的类
getSimpleName()获取被代理的目标对象的简单类名(不含包名)
getArgs()获取方法参数数组
getSignature()获取方法签名,包括方法名、返回类型、参数类型等信息
getModifiers()获取方法修饰符

有三个案例分别是一般情况,需要返回值情况,还有异常情况

一般情况(前置通知、后置通知)

案例代码:

需要在方法调用中参数加入JoinPoint接口实例化对象用以创建对应的动态代理,并通过动态代理获取对象相关信息。

public class advice {


    @Before("execution(* com.alphamilk.*.*(..))")
    public void Before(JoinPoint joinPoint) {
//        获取类名
        String simpleName = joinPoint.getTarget().getClass().getSimpleName();
//        获取方法修饰符
        int modifiers = joinPoint.getSignature().getModifiers();
        String Moidfier = Modifier.toString(modifiers);
//        获取方法名称
        String name = joinPoint.getSignature().getName();
//        获取参数
        Object[] args = joinPoint.getArgs();
//
        System.out.println("调用的方法是" + name);
        System.out.println("调用的类是" + simpleName);
        for (Object arg : args
        ) {
            System.out.println(arg);
        }
        System.out.println("调用方法前");
    }

    @After("execution(* com.alphamilk.*.*(..))")
    public void After(JoinPoint joinPoint) {
        System.out.println("调用方法后");
    }
}

有返回值的情况(返回通知)

在一般情况的前提下,还需要多增加Object result参数用以接收返回值.和注解增加returning输入确切的返回对象的名称。

案例代码

public class advice {

    @AfterReturning(value = "execution(* com.alphamilk.*.*(..))",returning = "result")
    public void AfterReturning(JoinPoint joinPoint,Object result) {
        System.out.println("调用拥有返回值的方法");
        System.out.println("获取到的返回值为"+result);
    }
}

异常情况(异常通知)

异常通知,获取异常信息,需要在一般情况的前提下,在注解中多声明一个注解throwing,在方法参数增加一个Throwable对象,并且throwing注解对应的值就是Throwable的对象名称。

案例代码:

 @AfterThrowing(value = "execution(* com.alphamilk.*.*(..))",throwing = "throwable")
    public void AfterThrowing(JoinPoint joinPoint,Throwable throwable) {
        System.out.println("调用有异常的方法");
        System.out.println("异常对象为"+throwable.getClass().getName());
    }
@SpringJUnitConfig(value = JavaConfig.class)
public class newaopTest {
    @Autowired
    private Caculate caculate;
    @Test
    public void Test(){
        caculate.div(2,0);
    }
}


四、 切点表达式语法

1.切点表达式作用

AOP切点表达式(Pointcut Expression)是一种用于指定切点的语言,它可以通过定义匹配规则,来选择需要被切入的目标对象。

2.切点表达式语法

  • 具体值:

    • (String, int):第一个参数是字符串,第二个参数是整数。
    • (int, String):第一个参数是整数,第二个参数是字符串。
    • ():没有参数。
  • 模糊值:

    • (..):任意参数,有或者没有。
  • 部分具体和模糊:

    • (String..):第一个参数是字符串,后面可能有其他参数。
    • (..String):最后一个参数是字符串,前面可能有其他参数。
    • (String..int):字符串开头,最后一个参数是整数,中间可能有其他参数。
    • (..int..):包含整数类型的参数,位置不限,可能有其他参数。

具体实战案例:

1.查询某包某类下,访问修饰符是公有,返回值是int的全部方法

execution public int 某包.某类.*(..)

2.查询某包下类中第一个参数是String的方法

execution * 某包.某类.*(String..)

3.查询全部包下,无参数的方法!

execution * *..*.*( )

4.查询com包下,以int参数类型结尾的方法

execution * com..*.*(..int)

5.查询指定包下,Service开头类的私有返回值int的无参数方法

execution private int 指定包.Service*.*()


五、重用(提取)切点表达式

如果在每一个方法前都加上一个固定的切点表达式,那么将会十分麻烦,所以下面介绍切点表达式的重用

1.在当前类中提取

特定注解@Pointcut

注解描述
@Pointcut声明切点表达式的方法,用于定义切点的匹配规则。

通过定义一个空方法,使用@Pointcut注解并带上特定的切点表达式

案例代码:

@Component
@Aspect
public class advice {
    
    /*
    定义空方法
    空方法上加上注解@Pointcut并带上相应的切点表达式
    在其他增强方法上调用方法
     */
    @Pointcut("execution(* com.alphamilk.*.*(..))" )
    public void blank(){}

  @Before("blank()")
    public void Before(JoinPoint joinPoint) {
        System.out.println("调用方法前");
    }

 @After("blank()")
    public void After(JoinPoint joinPoint) {
        System.out.println("调用方法后");
    }

    @AfterReturning(value = "blank()",returning = "result")
    public void AfterReturning(JoinPoint joinPoint,Object result) {
        System.out.println("调用拥有返回值的方法");
    }

    @AfterThrowing(value = "blank()",throwing = "throwable")
    public void AfterThrowing(JoinPoint joinPoint,Throwable throwable) {
        System.out.println("调用有异常的方法");
    }
}

2.创建一个存储切点类

(推荐)通过创建一个单独的存储切点的类,更加容易进行维护表达式

  使用时候加上特定类的方法名即可

案例:

存储切点的类

@Component
public class MyPointcut {
    @Pointcut("execution(* com.alphamilk.Impl.*.*(..))")
    public void pointcut1(){}
}

对应引用类

@Component
@Aspect
public class advice {
  @Before("com.alphamilk.Advice.MyPointcut.pointcut1()")
    public void Before(JoinPoint joinPoint) {
        System.out.println("调用方法前");
    }

 @After("com.alphamilk.Advice.MyPointcut.pointcut1()")
    public void After(JoinPoint joinPoint) {
        System.out.println("调用方法后");
    }

    @AfterReturning(value = "com.alphamilk.Advice.MyPointcut.pointcut1()",returning = "result")
    public void AfterReturning(JoinPoint joinPoint,Object result) {
        System.out.println("调用拥有返回值的方法");
    }

    @AfterThrowing(value = "com.alphamilk.Advice.MyPointcut.pointcut1()",throwing = "throwable")
    public void AfterThrowing(JoinPoint joinPoint,Throwable throwable) {
        System.out.println("调用有异常的方法");
    }
}

本章总结

1.SpringAop底层技术

        了解底层代理技术有jdk 与 cglib

2.初步实现AOP编程

        掌握增强注解(@Before、@AfterReturning、@AfterThrowing、@After、@Around)

        掌握@Aspect注解的使用

3.获取切点详细信息

        掌握如何通过JoinPoint接口对象获取对应方法的类,方法名称,参数,方法修饰符

        掌握三种不同情况下获取对应信息的情况(一般情况、返回通知、异常通知)

4.切点表达式语法

        熟悉切点表达式的格式

      (execution +权限修饰 +方法返回值类型+方法所在全类名+方法名称+参数列表)

5.重用(提取)切点表达式

        

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/938324.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

国产AI芯片突破,芯片或成白菜价,恐惧的美芯阻止台积电为它代工

日前消息指台积电大幅减少一家中国AI芯片企业的产能&#xff0c;原因在于国产AI芯片的性能已接近美芯&#xff0c;美国芯片企业NVIDIA与相关的资本机构贝莱德联手施压台积电所致&#xff0c;凸显出美国芯片忧虑中国AI芯片的竞争力。 这家国产AI芯片企业为壁仞科技&#xff0c;据…

C#,《小白学程序》第七课:列表(List)应用之一————编制高铁车次信息表

1 文本格式 /// <summary> /// 车站信息类 class /// </summary> public class Station { /// <summary> /// 编号 /// </summary> public int Id { get; set; } 0; /// <summary> /// 车站名 /// </summary>…

【JavaSE专栏89】Java字符串和XML数据结构的转换,高效灵活转变数据

作者主页&#xff1a;Designer 小郑 作者简介&#xff1a;3年JAVA全栈开发经验&#xff0c;专注JAVA技术、系统定制、远程指导&#xff0c;致力于企业数字化转型&#xff0c;CSDN学院、蓝桥云课认证讲师。 主打方向&#xff1a;Vue、SpringBoot、微信小程序 本文讲解了 XML 的概…

软件测试的方法有哪些?

软件测试 根据利用的被测对象信息的不同&#xff0c;可以将软件测试方法分为&#xff1a;黑盒测试、灰盒测试、白盒测试。 1、白盒测试 1&#xff09;概念&#xff1a;是依据被测软件分析程序内部构造&#xff0c;并根据内部构造分析用例&#xff0c;来对内部控制流程进行测试…

基于Dpabi的功能连接

1.预处理 这里预处理用Gretna软件进行&#xff0c;共分为以下几步&#xff1a; &#xff08;1&#xff09;DICOM转NIfTI格式 (2)去除前10个时间点(Remove first 10 times points)&#xff1a;由于机器刚启动、被试刚躺进去也还需适应环境&#xff0c;导致刚开始扫描的数据很…

macOS 安装 Homebrew 详细过程

文章目录 macOS 安装 Homebrew 详细过程Homebrew 简介Homebrew 安装过程设置环境变量安装 Homebrew安装完成后续设置(重要)设置环境变量homebrew 镜像源设置macOS 安装 Homebrew 详细过程 本文讲解了如何使用中科大源安装 Homebrew 的安装过程,文章里面的所有步骤都是必要的,需…

ExpressLRS开源之RC链路性能测试

ExpressLRS开源之RC链路性能测试 1. 源由2. 分析3. 测试方案4. 测试设计4.1 校准测试4.2 实验室测试4.3 拉距测试4.4 遮挡测试 5. 总结6. 参考资料 1. 源由 基于ExpressLRS开源基本调试验证方法&#xff0c;对RC链路性能进行简单的性能测试。 修改设计总能够满足合理的需求&a…

Streamlit 讲解专栏(十一):数据可视化-图表绘制详解(中)

文章目录 1 前言2 绘制交互式散点图3 定制图表主题4 增强数据可视化的交互性与注释步骤1步骤二 5 结语 1 前言 在上一篇博文《 Streamlit 讲解专栏&#xff08;十&#xff09;&#xff1a;数据可视化-图表绘制详解&#xff08;上&#xff09;》中&#xff0c;我们学习了一些关…

Vue脚手架中安装ElementUi

目录 ElementUi简介&#xff1a; ElementUi下载&#xff1a; npm 安装&#xff1a; 引入ElementUi: 测试是否引入成功&#xff1a; Element-ui官网&#xff1a;组件 | Element ElementUi简介&#xff1a; ElementUi&#xff0c;是由国内的饿了么团队开发并开源的一套为开…

美五代机装备激光武器可行性分析

源自&#xff1a;北京蓝德信息科技有限公司 一、SHiELD项目研究进展分析 图表&#xff1a;SHiELD项目主要情况 二、机载激光武器面临的技术挑战分析 三、五代机装备激光武器的可行性 声明:公众号转载的文章及图片出于非商业性的教育和科研目的供大家参考和探讨&#xff0c;并不…

三维模型OBJ格式轻量化压缩处理效率提高的技术方法探讨

三维模型OBJ格式轻量化压缩处理效率提高的技术方法探讨 要提高三维模型OBJ格式轻量化压缩处理的效率&#xff0c;可以采取以下方法&#xff1a; 1、优化算法选择&#xff1a;选择合适的优化算法对模型进行轻量化处理。不同的优化算法有不同的时间复杂度和效果。一些常用的优化…

软件测试用例经典方法 | 因果图法及案例

典型的黑盒测试用例设计方法包括等价类划分法、边界值分析法、决策表法、因果图法等。 如果程序的输入条件之间相互存在联系,那么就会使情况变得复杂,因为要检查输入条件的组合情况并不是一件容易的事情,即使把所有输入条件划分为等价类,它们之间的组合情况也相当多,难以分析。…

【Go 基础篇】深入探索:Go语言中的二维数组

在计算机编程中&#xff0c;数组是一种基本的数据结构&#xff0c;用于存储相同类型的元素。而二维数组作为数组的一种扩展&#xff0c;允许我们以类似表格的方式存储和处理数据。在Go语言中&#xff0c;二维数组是一个重要的概念&#xff0c;本文将深入探讨Go语言中的二维数组…

兄弟,王者荣耀的段位排行榜是通过Redis实现的?

目录 一、排行榜设计方案1、数据库直接排序2、王者荣耀好友排行 二、Redis实现计数器1、什么是计数器功能&#xff1f;2、Redis实现计数器的原理&#xff08;1&#xff09;使用INCR命令实现计数器&#xff08;2&#xff09;使用INCRBY命令实现计数器 三、通过Redis实现“王者荣…

基于知识引入的情感分析研究综述

1.引文 情感分析知识 当training数据不足以覆盖inference阶段遇到的特征时&#xff0c;是标注更多的数据还是利用现有外部知识充当监督信号&#xff1f; 基于机器学习、深度学习的情感分析方法&#xff0c;经常会遇到有标注数据不足&#xff0c;在实际应用过程中泛化能力差的局…

C语言巧用联合体union判定数据的存储格式(大小端)

联合体大家可能比较陌生&#xff0c;但是大家对结构体稍微熟悉一点吧。其实它们二个类似&#xff0c;只不过结构体成员占用不同的地址&#xff0c;而联合体所有成员占用相同地址。利用这个特性我们就能判断在当前编译器下存储的数据的格式。那么如何确定呢&#xff1f; 我这里…

FPGA GTX全网最细讲解,aurora 8b/10b协议,OV5640板对板视频传输,提供2套工程源码和技术支持

目录 1、前言免责声明 2、我这里已有的 GT 高速接口解决方案3、GTX 全网最细解读GTX 基本结构GTX 发送和接收处理流程GTX 的参考时钟GTX 发送接口GTX 接收接口GTX IP核调用和使用 4、设计思路框架视频源选择OV5640摄像头配置及采集动态彩条视频数据组包GTX aurora 8b/10b数据对…

Java“牵手”天猫图片识别商品信息API接口数据,图片搜索商品接口,天猫拍立淘API接口申请指南

天猫平台按图搜商品接口&#xff08;拍立淘&#xff09;是开放平台提供的一种API接口&#xff0c;通过调用API接口&#xff0c;开发者可以获取天猫商品的标题、价格、库存、月销量、总销量、库存、详情描述、图片、最低价、当前价格、价格信息等详细信息 。 获取拍立淘接口API…

Spring Boot Actuator的Env端点存在本地文件包含(LFI)漏洞CVE-2020-5421

文章目录 0.前言1.参考文档2.基础介绍3.漏洞利用原理3.解决方案1. 升级Spring Boot版本2. 限制端点的访问3. 禁用环境端点4. 不公开敏感的Actuator端点5. 开启安全审计 0.前言 背景&#xff1a; Spring Boot Actuator的Env端点存在本地文件包含(LFI)漏洞CVE-2020-5421。被扫描到…

软件工程(二十) 系统运行与软件维护

1、系统转换计划 1.1、遗留系统的演化策略 时至今日,你想去开发一个系统,想完全不涉及到已有的系统,基本是不可能的事情。但是对于已有系统我们有一个策略。 比如我们是淘汰掉已有系统,还是继承已有系统,或者集成已有系统,或者改造遗留的系统呢,都是不同的策略。 技术…