Springboot AOP开发

Springboot AOP开发

一 AOP概述

AOP,即面向切面编程,简言之,面向方法编程。

针对方法,在方法的执行前或执行后使用,用于增强方法,或拓展。

二 AOP开发

1.引入 spring-boot-starter-aop

在SpringBoot项目的pom文件中,引入 spring-boot-starter-aop依赖。

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId>
</dependency>

2.示例:计算方法执行时间

1.创建实体类,通过注解来申明该类的类型,并将该类交给Spring的IOC容器来管理。

通过注解 @Aspect 申明这是一个AOP类

通过 @Component 将其交给IOC容器管理

@Aspect
@Component
public class TimeAspect {//code
}

2.创建方法并且实现

@Aspect
@Component
@Slf4j
public class TimeAspect {// 针对 com.shawn.springboot03.service 包下所有的方法进行编程,// * com.shawn.springboot03.service.*.*(..)) 为切入点表达式@Around("execution(* com.shawn.springboot03.service.*.*(..))")public Object recordTime(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {long startTime=System.currentTimeMillis();// 切面对象,执行具体的业务方法Object object = proceedingJoinPoint.proceed();long endTime=System.currentTimeMillis();log.info("方法耗时为{}ms",endTime-startTime);return object;}
}

通过以上方法,当用户调用 service 层接口的任一方法时都会计算方法的运行时间。

3.AOP编程的优点:

  • 代码无侵入:无需修改原始方法
  • 减少代码重复,提高开发效率:只需编写一次
  • 维护方便:根据业务需求,调整切入点表达式即可

三 AOP详解

1.AOP核心概念

1.连接点(JoinPoint):连接点指的是可以被AOP控制的方法,以及方法执行时的相关信息。

2.通知(Advice):Advice指的是被抽取出来的共性功能,即重复的那部分逻辑。

3.切入点(PointCut):匹配连接点的条件,通知仅会在切入点方法执行时被应用

4.切面(Aspcet):描述通知与切入点的关系

5.目标对象(Target):通知所应用的对象

2.AOP通知类型

  • @Around:环绕通知,此注解标注的方法在目标方法前后都会执行
  • @Before:前置通知,此注解标注的方法仅在方法执行前被执行
  • @After:后置通知,此注解在方法执行完成后执行,不论是否抛出异常
  • @AfterReturning:返回后通知,此注解标注的方法在目标方法后被执行,有异常不通知
  • @AfterThrowing:异常后通知,此注解的通知方法发生异常后执行

3.各通知类型演示

创建一个 TestAspect 类,用于演示各种通知类型

  1. @Around 通知类型

    介绍:在方法前后均执行

    切入点表达式格式: 返回值 包名.方法名(形参)

    其中 * 代表全部,.. 代表任意多的参数

    切入点表达式示例说明:

    * com.shawn.test.server.*(..) 表示 任意返回值的 com.shawn.test.server包下任意返回值,任意多参数的全部方法。

    @Aspect
    @Component
    @Slf4j
    public class TestAspect {@Around("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))")public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {//方法执行前的业务逻辑//code..//执行原始方法(即连接点),并且接收原始方法的返回值Object proceed = proceedingJoinPoint.proceed();//方法执行后的业务逻辑//code..//返回原始方法的返回值return proceed;}
    }
    

    ProceedingJoinPoint是一个接口类,指代程序执行过程中的一个特定点,其中包含了原始方法的全部内容,包括方法名,参数值等。

    在使用 @Around通知类型时,需要将该对象作为参数传递进来,用于后续执行原始方法或获取原始方法的其他信息。

    ProceedingJoinPoint仅能作用于 @Around 类型的通知上。

2.@Before 通知类型

介绍:此注解标注的方法仅在原始方法执行前执行

@Aspect
@Component
@Slf4j
public class TestAspect {@Before("execution(* com.shawn.springboot03.service.*.*(..))")public void before(){// 方法执行前的业务逻辑//code..}
}

3.@After 通知类型

介绍:此注解标注的方法,仅在原始方法执行后执行,且不论原始方法是否执行,该通知都会执行

@Aspect
@Component
@Slf4j
public class TestAspect {@After("execution(* com.shawn.springboot03.service.*.*(..))")public void after(){// 方法执行后的业务逻辑//code..}
}

4.@AfterReturning 通知类型

介绍:此注解标注的方法,仅在原始方法执行完成后通知,即当原始方法执行发生异常时,@AfterReturning 通知不会被执行。

@Aspect
@Component
@Slf4j
public class TestAspect {@AfterReturning("execution(* com.shawn.springboot03.service.*.*(..))")public void afterReturning(){//方法执行完成的业务逻辑//code..}
}

5.@AfterThrowing通知类型

介绍:次注解标注的方法,仅在原始方法执行过程中发生了异常,才会执行。

@Aspect
@Component
@Slf4j
public class TestAspect {@AfterThrowing("execution(* com.shawn.springboot03.service.*.*(..))")public void afterThrowing(){//方法执行时抛出异常的业务逻辑//code..}
}

4.定义切入点

以上示例中的切入点表达式均相似,可以利用封装的思想,将切入点表达式全部抽取出来,在需要的时候直接使用即可。

想要实现以上需求,则需要自定义切入点表达式

通过 @Pointcut 注解,来定义切入点表达式,然后在需要编写切入点表达式的地方调用即可。

@Aspect
@Component
@Slf4j
public class TestAspect {/*** 声明一个空的方法体来定义切入点表达式* 使用 @Pointcut 注解来定义切入点表达式*/@Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))")private void point(){};@After("point()")public void after(){// 方法执行后的业务逻辑//code..}
}

注意:如果自定义的切入点访问修饰符为 public ,则该表达式还可以在其他切面类中被引用。具体使用可根据实际业务情况决定。

定义切入点总共有两种办法,一种是通过方法名来定义切入点,还可以通过注解来定义切入点。

示例:以下示例使用自定义注解作为切入点,使用了以下自定义注解的方法会执行通知。

@Aspect
@Component
public class TestAspect {@Pointcut("@annotation(com.shawn.springboot03.annotation.OperationLog)")private void point(){};@AfterThrowing("point()")public void afterThrowing(){//方法执行时抛出异常的业务逻辑//code..}
}

5.通知顺序

当有多个切入点都匹配到了目标方法,目标方法运行时,多个通知都会被执行。

某些情况下则需要控制通知的顺序,可以在切面类上使用 @Order 注解来控制顺序

@Aspect
@Component
@Slf4j
@Order(3)
public class TestAspect {@Pointcut("execution(* com.shawn.springboot03.service.impl.DeptServiceImpl.*(..))")private void point(){};@After("point()")public void after(){// 方法执行后的业务逻辑//code..}
}

注意:@Order(number) 注解中,数字越小的越先运行,越大的越后运行。

默认情况下,按照切面类的类名排序顺序执行。

四 切入点表达式

1.execution

execution 主要根据方法的返回值,包名类名,方法名,方法参数等信息来匹配,语法为:

execution( 访问修饰符? 返回值 包名.类名.?方法名(方法参数) throw 异常? )

其中带 ? 的表示可以省略的部分

  • 访问修饰符:可省略(比如:public,protected)
  • 包名.类名:可省略
  • throws 异常:可省略(注意是方法上声明抛出的异常,不是实际抛出的异常)

以下是一个完整示例:

@Before ("execution(public void com.itheima.service.impl.DeptserviceImpl.delete (java.lang.Integer)) throws Exception")
private void point(){};

省略后的示例:

@Before ("execution(void delete (java.lang.Integer))")
private void point(){};

此时,所有 void delete (java.lang.Integer) 方法都将被匹配到。

注意: * 用于描述匹配单个, … 可以用来匹配多个连续,?表示可省略
在这里插入图片描述

2.@annotation

@annotation 切入点表达式,用于匹配标识有特定注解的方法。

示例:

@Pointcut("@annotation(com.shawn.annotation.OperationLog)")
private void point(){};

五 连接点

在Spring中使用JoinPoint抽象了连接点,使用它可以获取方法执行时的相关信息,入目标类名,方法名,方法参数等。

1.对于 @Around 通知,获取连接点信息只能使用 ProceedingJoinPoint

@Around("point()")
public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {//获取目标类名String name = proceedingJoinPoint.getTarget().getClass().getName();log.info("获取目标类名:"+name);//获取方法执行参数Object[] args = proceedingJoinPoint.getArgs();log.info("获取方法执行参数:"+JSON.toJSONString(args));//获取目标方法名String methodName = proceedingJoinPoint.getSignature().getName();log.info("获取目标方法名:"+methodName);//执行原始方法(即连接点),并且接收原始方法的返回值Object proceed = proceedingJoinPoint.proceed();//获取方法的返回值log.info("获取方法的返回值:"+JSON.toJSONString(proceed));//返回原始方法的返回值return proceed;
}

proceedingJoinPoint.proceed()得到目标方法的结果再返回,此处可以对目标方法执行的结果进行篡改。

2.对于其他四种通知,获取连接点信息只能使用 JoinPoint , 它是 ProceedingJoinPoint 的父类。

@Before("point()")
public void before(JoinPoint joinPoint){// 获取目标方法的类名String name = joinPoint.getTarget().getClass().getName();log.info("获取目标方法的类名:"+name);String methodName = joinPoint.getSignature().getName();log.info("获取目标方法的方法名:"+methodName);//获取目标方法的运行参数Object[] args = joinPoint.getArgs();log.info("获取目标方法的参数:"+JSON.toJSONString(args));
}

3.除@Around 通知外,还可以获取到方法执行结果的通知类型为 @AfterReturning,@AfterReturning 在方法执行完成并且无异常时通知。

在定义@AfterReturning通知类型时,使用 pointcut 属性定义切入点,returning属性定义返回值对象,然后在方法参数中传入即可。

@AfterReturning(pointcut = "point()",returning = "object")
public void afterReturning(JoinPoint joinPoint, Object object){// 获取目标方法的类名String name = joinPoint.getTarget().getClass().getName();log.info("获取目标方法的类名:"+name);String methodName = joinPoint.getSignature().getName();log.info("获取目标方法的方法名:"+methodName);//获取目标方法的运行参数Object[] args = joinPoint.getArgs();log.info("获取目标方法的参数:"+JSON.toJSONString(args));//获取方法返回值log.info("获取方法返回值:"+JSON.toJSONString(object));
}

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

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

相关文章

AWS Elastic Beanstalk通过应用负载均衡配置https

接上一篇&#xff0c;今天说说怎么通过AWS Elastic Beanstalk提供的应用负载均衡配置https。 首先创建应用和环境&#xff0c;这里应用可以使用上一篇文章中使用的demo应用&#xff08;只需要package.json和app.js文件&#xff09; 创建环境的时候&#xff0c;确认下面两个参…

【高效开发工具系列】PyCharm使用

&#x1f49d;&#x1f49d;&#x1f49d;欢迎来到我的博客&#xff0c;很高兴能够在这里和您见面&#xff01;希望您在这里可以感受到一份轻松愉快的氛围&#xff0c;不仅可以获得有趣的内容和知识&#xff0c;也可以畅所欲言、分享您的想法和见解。 推荐:kwan 的首页,持续学…

自养号测评低成本高效率推广,安全可控

测评的作用在于让用户更真实、清晰、快捷地了解产品以及产品的使用方法和体验。通过买家对产品的测评&#xff0c;也可以帮助厂商和卖家优化产品缺陷&#xff0c;提高用户的使用体验。这进而帮助他们获得更好的销量&#xff0c;并更深入地了解市场需求。因此&#xff0c;测评在…

基于单片机的智能宠物喂食器设计

摘要:阐述智能宠物喂食器的实现方式,以STC89C52单片机为核心芯片,控制LCD的显示、语音芯片的启动和步进电机的运行。通过按键设置预设时间,当时间到达预设时间时,语音电路发出提示,步进电机工作,提供食物。此系统解决了主人由于各种原因不在家,使得宠物不能按时吃饭的问…

ACL 2024系统

文章目录 ACL官方网站信息通知ACL2024 官方模版 ACL官方网站信息通知 https://2024.aclweb.org/ ACL2024 官方模版 https://github.com/acl-org/acl-style-files

【Java中23种设计模式-单例模式2--懒汉式线程不安全】

加油&#xff0c;新时代打工人&#xff01; 今天&#xff0c;重新回顾一下设计模式&#xff0c;我们一起变强&#xff0c;变秃。哈哈。 23种设计模式定义介绍 Java中23种设计模式-单例模式 package mode;/*** author wenhao* date 2024/02/19 09:16* description 单例模式--懒…

.NET高级面试指南专题九【 泛型概念,常用泛型类和方法,泛型约束,协变与逆变】

C#中的泛型&#xff08;Generics&#xff09;是一种强大的编程特性&#xff0c;它允许你在编写代码时使用不特定数据类型&#xff0c;而在编译时确定这些数据类型。泛型使得代码更加灵活、可重用&#xff0c;并提高了类型安全性。 功能和原理 泛型允许你编写能够与不同数据类型…

数字孪生核心技术揭秘(五):BIM究竟是解药还是毒药?

一、关于BIM的那些幻想 1.1 BIM是“数字孪生城市”最后一块拼图&#xff1f; 近几年&#xff0c;CIM概念越来越流行&#xff0c;已经成为数字孪生城市的主流数据模型。CIM原始概念指的是“城市信息模型”&#xff1b;2015年同济大学吴志强院士基于CIM概念提出“城市智慧模型”&…

如何用ChatGPT绘图?

详情点查看公众号&#xff1a;技术科研吧 链接&#xff1a;如何用ChatGPT绘图&#xff1f; 一&#xff1a;AI领域最新技术 1.OpenAI新模型-GPT-5 2.谷歌新模型-Gemini Ultra 3.Meta新模型-LLama3 4.科大讯飞-星火认知 5.百度-文心一言 6.MoonshotAI-Kimi 7.智谱AI-GLM-…

23种设计模式-Golang(完整版)

23种设计模式-Golang 完整代码链接&#xff1a;https://github.com/ziyifast/easy_design_mode &#x1f680;&#xff1a;欢迎star哦&#xff5e; 1 创建型模式 1.1 简单工厂模式&#xff08;simple factory&#xff09;&#xff1a;不同协议有生成不同downloader ①解析 go…

ChatGPT在数据分析学习阶段的应用

ChatGPT在数据分析学习阶段的应用 ​ 这个阶段&#xff0c;核心是三件事&#xff1a;制定学习计划、确定学习资料以及学习策略。我们可以自己完成这几件事&#xff0c;当然也可以借助ChatGPT来高效地达到目的。 1.1 制定学习计划 ​ 学习阶段的第一件事是制定学习计划&#…

USACO 2024年1月铜组 MAJORITY OPINION

第一题&#xff1a;MAJORITY OPINION 标签&#xff1a;思维、模拟 题意&#xff1a;给定一个长度为 n n n的序列 a a a&#xff0c;操作&#xff1a;若区间 [ i , j ] [i,j] [i,j]内某个数字 k k k出现的次数 大于区间长度的一半&#xff0c;可以将区间内的所有数都换成这个数…

python之元类

最近刚好在准备面试&#xff0c;然后复习知识点。看了好多元类相关的文章&#xff0c;但还是心存疑惑&#x1f914;。 还是写点东西吧&#xff01; 元类&#xff0c;官方的定义是&#xff0c;类的类型。而类型的顶点&#xff0c;便是type metaclass - The class of a class.…

Java流程控制

1. 流程控制 在一个程序中&#xff0c;有很多的代码&#xff0c;代码的执行顺序&#xff0c;就是流程。 用于控制代码流程的方式&#xff0c;就是流程控制 流程控制的分类&#xff1a; 顺序结构&#xff1a;代码从上到下&#xff0c;从右到左依次执行。 分支结构&#xff1a…

OpenAI的Sora的优点

OpenAI的Sora的优点主要包括以下几个方面&#xff1a; 1.强大的视频生成能力&#xff1a;Sora可以根据用户的文本提示或静态图像生成高质量、逼真的视频。它继承了DALL-E 3的画质和遵循指令能力&#xff0c;能够生成具有多个角色、包含特定运动的复杂场景&#xff0c;几乎达到…

Unity3d Mesh篇(一)— 创建简单三角面

文章目录 前言一、Mesh组成二、使用步骤三、效果四、总结 前言 Mesh&#xff08;网格&#xff09;是一种常用的3D图形表示方法&#xff0c;它由顶点&#xff0c;法线&#xff0c;UV 坐标&#xff0c;和三角形等组成。您可以使用 Mesh 类的方法来创建或修改网格&#xff0c;也可…

k8s学习(RKE+k8s+rancher2.x)成长系列之简配版环境搭建(三)

3.19.切换RKE用户&#xff0c;并做免密登录&#xff08;三台机器相互免密&#xff09; su rke cd~ ssh-keygen[rkemaster.ssh]$ssh-copy-id rkeslaver2 [rkemaster.ssh]$ssh-copy-id rkeslaver1 [rkemaster.ssh]$ssh-copy-id rkemaster3.20.搭建RKE集群 为了方便理解&#…

office的excel中使用,告诉我详细的解决方案,如何变成转化为金额格式

在Office的Excel中&#xff0c;如果你想将名为"MEREFIELD"的公式结果转换为金额格式&#xff0c;你可以遵循以下详细步骤来实现&#xff1a; 书写MEREFIELD公式&#xff1a; 首先&#xff0c;在Excel中输入或确认你的MEREFIELD公式。例如&#xff0c;假设这个公式是用…

基于STL的演讲比赛流程管理系统(个人学习笔记黑马学习)

1、演讲比赛程序需求 1.1比赛规则 学校举行一场演讲比赛&#xff0c;共有12个人参加。比赛共两轮&#xff0c;第一轮为淘汰赛&#xff0c;第二轮为决赛。每名选手都有对应的编号&#xff0c;如 10001~10012比赛方式:分组比赛&#xff0c;每组6个人;第一轮分为两个小组&#xff…

SpringBoot 条件注解一览无余

文章目录 条件注解是什么有哪些条件注解类条件注解Bean条件注解属性条件注解资源条件注解web应用条件注解SpEL( Spring Expression Language )表达式条件注解其他条件注解 总结 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍…