Spring Boot学习随笔- 实现AOP(JoinPoint、ProceedingJoinPoint、自定义注解类实现切面)

学习视频:【编程不良人】2021年SpringBoot最新最全教程

第十一章、AOP

11.1 为什么要使用AOP

  • 问题

    1. 现有业务层开发存在问题
      1. 额外功能代码存在大量冗余
      2. 每个方法都需要书写一遍额外功能代码不利于项目维护
  • Spring中的AOP

    AOP:Aspect 切面 + Oriented 面向 Programmaing 面向切面编程

    Aspect(切面) = Advice(通知) + Pointcut(切入点)

    Advice 通知:业务逻辑中的一些附加操作称之通知

    Pointcut 切入点:配置通知应用于项目中那些业务操作

  • Advice通知就是附加操作的代码,Advice通知类型都有不同的执行策略和用途。

    @Before在目标方法执行之前执行的通知。它不能阻止方法的执行,但可以在方法执行前添加额外的功能。
    @AfterReturning在目标方法正常返回后执行的通知。例如,如果一个方法正常返回而没有抛出异常,就会执行这个通知。
    @AfterThrowing在目标方法抛出异常后执行的通知。如果一个方法抛出异常,就会执行这个通知。
    @After在目标方法执行之后执行的通知。无论目标方法如何退出(正常返回或抛出异常),都会执行的通知。
    @Around包围目标方法的通知,可以在目标方法执行前后添加额外的功能,并决定是否继续执行目标方法。
  • Pointcut 是切入点,决定了Advice加在哪个具体方法代码上,具体使用方式:

    1.切入点直接写在附加操作里面
    @Around(value="execution(* login(..))")
    2.通过@Pointcut注解 声明切入点,实现复用
    @Pointcut("execution(* login(..))") // 复用切入点,解耦合public void myPointcut(){}@Around("myPointcut()")
    @After("myPointcut()")
    @Before("myPointcut()")
    

11.2 AOP的实现

  • 引入依赖

    <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId>
    </dependency>
    
  • 目前UserServiceImp存在的问题

    以log日志这种额外功能为例,这样的重复操作存在冗余代码和耦合

    @Service
    public class UserServiceImpl implements UserService {@Overridepublic void save(String name) {System.out.println("========log========");System.out.println("UserServiceImpl.save");}@Overridepublic void delete(Integer id) {System.out.println("========log========");System.out.println("UserServiceImpl.delete");}@Overridepublic void update(String name) {System.out.println("========log========");System.out.println("UserServiceImpl.update");}@Overridepublic String find(String name) {System.out.println("========log========");System.out.println("UserServiceImpl.find");return "name";}
    }
    
  • 我们通过一个切面配置类来解耦合

    /*** 自定义切面配置类*/
    @Configuration  // 指定当前类为配置类
    @Aspect // 代表这个类是一个切面配置类
    public class MyAspectConfig {@Before("execution(* com.example.service.*.*(..))") // @Before代表在业务逻辑执行前运行 value代表切入点public void before() {System.out.println(" =====前置附加操作:log====== ");}
    }
    
  • 通过切入点,完成整个业务的额外功能覆盖,运行效果

11.3 JoinPoint 参数详解

**JoinPoint参数可以在通知体内声明,用于获取有关方法执行的信息。JoinPoint参数提供了许多有用的方法,例如getSignature()可以获取方法的签名,getArgs()可以获取方法的参数列表,getTarget()可以获取目标对象等。通过JoinPoint**参数,我们可以在通知中访问和操作方法执行时的上下文信息。

**JoinPoint**参数通常用于以下几种情况:

  1. 记录日志:获取方法的签名和参数列表,从而记录方法的执行情况,包括方法名、参数值等。
  2. 异常处理:获取方法执行时抛出的异常信息,从而进行相应的异常处理。
  3. 性能监控:获取方法的执行时间、参数值等信息,用于性能监控和优化。
获取执行方法的信息
@Before("execution(* com.example.service.*.*(..))") // 
public void before(JoinPoint joinPoint) {String methodName = joinPoint.getSignature().getName();String className = joinPoint.getTarget().getClass().getName();Object[] args = joinPoint.getArgs();System.out.println("Before 执行方法: " + methodName + " 在类:" + className);System.out.println("方法参数 = " + Arrays.toString(args));
}

11.4 @Around 环绕附加操作

使用@Around注解的方法,参数必须声明ProceedingJoinPoint,这个参数可以控制目标方法的执行

@Around("execution(* com.example.service.*.*(..))")
public Object around(ProceedingJoinPoint pjp) throws Throwable {System.out.println("======进入环绕的前置操作======");System.out.println("当前执行类 = " + pjp.getSignature().getName());System.out.println("方法名 = " + pjp.getTarget().getClass().getSimpleName());//放行目标方法执行Object proceed = pjp.proceed(); // 目标方法继续执行System.out.println(" =====进入环绕的后置操作========= ");return proceed;
}
  • 运行效果

11.5 自定义注解方式的切入点表达式@annotation

通过自定义注解@annotation实现切面的好处在于可以使切面的定义更加灵活和可重用。使用自定义注解可以将切面的逻辑和配置信息封装在注解中,使得切面的使用和配置变得更加简单和直观。

  • MyAdvice 自定义注解

    @Retention(RetentionPolicy.RUNTIME) // 指定运行时保留
    @Target(ElementType.METHOD) //指定修饰 方法
    public @interface MyAdvice {
    }
    
  • 放在通知里的@annotation属性使用,代表只有被@MyAdvice修饰的方法才会被加入额外功能

    **@Around("@annotation(com.example.config.MyAdvice)")**
    public Object around(ProceedingJoinPoint pjp) throws Throwable {}
    
  • 通过将注解加在业务逻辑上,实现给目标方法加上额外功能的目的

    UserServiceImpl	
    @Override
    @MyAdvice
    public String find(String name) {
    //        System.out.println("========log========");System.out.println("UserServiceImpl.find");return "name";
    }
    

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

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

相关文章

竞赛保研 python 机器视觉 车牌识别 - opencv 深度学习 机器学习

1 前言 &#x1f525; 优质竞赛项目系列&#xff0c;今天要分享的是 &#x1f6a9; 基于python 机器视觉 的车牌识别系统 &#x1f947;学长这里给一个题目综合评分(每项满分5分) 难度系数&#xff1a;3分工作量&#xff1a;3分创新点&#xff1a;3分 &#x1f9ff; 更多资…

【Python 基础】-- 在 mac OS 中安装 多个 python 版本

目录 1、需求 2、实现 2.1 安装 pyenv 2.2 安装 pyenv-virtualenv 2.3 配置环境变量 2.4 创建 python 3.9.9 的环境 2.5 激活环境&#xff0c;在当前项目目录中使用&#xff0c;即执行 python 1、需求 由于项目所依赖的 python 版本有多个&#xff0c;需要在不同的 pyth…

主从reactor多线程实现

现场模型图片&#xff0c;从网上找的 出于学习的目的实现的&#xff0c;如有不对的地方欢迎留言知道&#xff0c;简单实现了http的请求&#xff0c;可通过postman进行访问 启动项目&#xff1a; 返回数据示例 postman请求 附上源码&#xff0c;有问题直接看源码吧

智能优化算法应用:基于闪电连接过程算法3D无线传感器网络(WSN)覆盖优化 - 附代码

智能优化算法应用&#xff1a;基于闪电连接过程算法3D无线传感器网络(WSN)覆盖优化 - 附代码 文章目录 智能优化算法应用&#xff1a;基于闪电连接过程算法3D无线传感器网络(WSN)覆盖优化 - 附代码1.无线传感网络节点模型2.覆盖数学模型及分析3.闪电连接过程算法4.实验参数设定…

Flink实时电商数仓(二)

GitLab的用户创建和推送 在root用户-密码界面重新设置密码添加Leader用户和自己使用的用户使用root用户创建相应的群组使用Leader用户创建对应的项目设置分支配置为“初始推送后完全保护”设置.gitignore文件&#xff0c;项目配置文件等其他非通用代码无需提交安装gitlab proj…

(JAVA)-创建多线程的方式

1.继承Thread类 1.创建一个继承字Thread类的子类 2.重写Thread类的run方法 public class MyThread extends Thread{Overridepublic void run() {for (int i 0; i < 100; i) {System.out.println(getName()"hello");}} }3.创建Thread类的子类对象 4.通过子类对象调…

HarmonyOS应用开发实战—开箱即用的应用首页页面【ArkTS】【鸿蒙专栏-34】

一.HarmonyOS应用开发实战—开箱即用的应用首页页面【ArkTS】【鸿蒙专栏-34】 1.1 项目背景 HarmonyOS(鸿蒙操作系统)是华为公司推出的一种分布式操作系统。它被设计为一种全场景、全连接的操作系统,旨在实现在各种设备之间的无缝协同和共享,包括智能手机、平板电脑、智能…

轻量级购物小程序H5产品设计经典样例

主要是看到这个产品设计的不错值得借鉴特记录如下&#xff1a; 不过大多数购物app都大致相同&#xff0c;这个算是经典样例&#xff0c;几乎都可以复制&#xff0c;我第一次使用&#xff0c;感觉和顺畅。看上去产品是经过打磨的&#xff0c;布局非常好。内容也很丰富。支持异业…

Leetcode—128.最长连续序列【中等】

2023每日刷题&#xff08;六十四&#xff09; Leetcode—128.最长连续序列 实现代码 class Solution { public:int longestConsecutive(vector<int>& nums) {unordered_set<int> s;for(auto num: nums) {s.insert(num);}int longestNum 0;for(auto num: s) …

LeetCode day27

LeetCode day27 —今天做到树&#xff0c;&#xff0c;&#xff0c;对不起我的数据结构老师啊~~~ 7. 整数反转 给你一个 32 位的有符号整数 x &#xff0c;返回将 x 中的数字部分反转后的结果。 如果反转后整数超过 32 位的有符号整数的范围 [−231, 231 − 1] &#xff0c…

【AI图集】猫狗的自动化合成图集

猫是一种哺乳动物&#xff0c;通常被人们作为宠物饲养。它们有柔软的毛发&#xff0c;灵活的身体和尖锐的爪子。猫是肉食性动物&#xff0c;主要以肉类为食&#xff0c;但也可以吃一些蔬菜和水果。猫通常在夜间活动&#xff0c;因此它们需要足够的玩具和活动空间来保持健康和快…

Golang(壹)

爱情不需要华丽的言语&#xff0c;只需要默默的行动。 简介 应用领域&#xff1a; 下载vscode 使用vscode Go下载 - Go语言中文网 - Golang中文社区 下载sdk 解压到文件中&#xff0c;打开sdk解压文件 穿插dos操作系统知识点&#xff1a; 测试go语言环境 看到vscode 的目录结…

SpringIOC之AnnotatedElementKey

博主介绍&#xff1a;✌全网粉丝5W&#xff0c;全栈开发工程师&#xff0c;从事多年软件开发&#xff0c;在大厂呆过。持有软件中级、六级等证书。可提供微服务项目搭建与毕业项目实战&#xff0c;博主也曾写过优秀论文&#xff0c;查重率极低&#xff0c;在这方面有丰富的经验…

Unity学习笔记(零基础到就业)|Chapter01:C#入门

Unity学习笔记&#xff08;零基础到就业&#xff09;&#xff5c;Chapter01:C#入门 前言一、控制台输入输出语句二、初识变量1.一些好用的tips2.变量声明的固定写法3.变量类型 三、变量的本质1.变量的存储空间2.变量的本质&#xff1a;2进制 四、变量的命名规范1.必须遵守的规则…

centos开机自启动实战小案例

1.编写一个我们需要做事的脚本 #!/bin/bash # 打印 "Hello" echo "Hello,Mr.Phor" # 为了更好的能看到效果 我们把这段文本放置到一个文件中 如果重启能够看到 /a.txt文件 我们实验成功 echo "hahahahahahahaha" > /a.txt #每次开机 执行…

《空气质量持续改善行动计划》发布,汉威科技助力蓝天保卫战

近日&#xff0c;国务院印发《空气质量持续改善行动计划》&#xff0c;这是继2013年“大气十条”之后的第三个国家层面的保卫蓝天行动计划。 计划要求协同推进降碳、减污、扩绿、增长&#xff0c;以改善空气质量为核心&#xff0c;以减少重污染天气和解决人民群众身边的突出大…

保护您的Android应用程序:Android应用程序安全一览

保护您的Android应用程序&#xff1a;Android应用程序安全一览 我们都知道Android是为所有人设计的——开放、面向开发者、面向用户&#xff0c;这种开放性为今天和明天的移动技术提供了很多便利。然而&#xff0c;开放性也带来了需要妥善处理的安全风险。 安全是我们所有人都…

WPF仿网易云搭建笔记(7):HandyControl重构

文章目录 专栏和Gitee仓库前言相关文章 新建项目项目环境项目结构 代码结果结尾 专栏和Gitee仓库 WPF仿网易云 Gitee仓库 WPF仿网易云 CSDN博客专栏 前言 最近我发现Material Design UI的功能比较简单&#xff0c;想实现一些比较简单的功能&#xff0c;比如消息提示&#xff0…

2018年第七届数学建模国际赛小美赛C题共享单车对城市交通的影响解题全过程文档及程序

2018年第七届数学建模国际赛小美赛 C题 共享单车对城市交通的影响 原题再现&#xff1a; 共享自行车改变了许多城市的交通状况&#xff0c;许多大城市引入共享自行车来解决交通问题。我们需要定量评估共享自行车对城市交通的影响&#xff0c;以及相关的经济、社会和环境影响。…

node.js mongoose中间件(middleware)

目录 简介 定义模型 注册中间件 创建doc实例&#xff0c;并进行增删改查 方法名和注册的中间件名相匹配 执行结果 分析 错误处理中间件 手动抛出错误 注意点 简介 在mongoose中&#xff0c;中间件是一种允许在执行数据库操作前&#xff08;pre&#xff09;或后&…