能免费做封面的网站/大兴今日头条新闻
目录
🚩Spring AOP概述
🚩Spring AOP快速⼊⻔
🎓引入AOP依赖
🎓编写AOP程序
🚩Spring AOP 详解
🎓Spring AOP核⼼概念
🎓通知类型
🎓@PointCut
🎓切⾯优先级 @Order
🎓切点表达式
📝execution表达式
📝@annotation
🎈⾃定义注解 @MyAspect
🎈切⾯类
🎈添加⾃定义注解
🚩Spring AOP的实现⽅式(常⻅⾯试题)
🚩Spring AOP概述
Spring框架的第一大核心是IOC(控制反转),接下来我们学习第二大核心——AOP.
什么是AOP?
Aspect Oriented Programming(⾯向切⾯编程)和面向对象编程不是互斥的关系,也不是面向对象编程的升级,是面向对象编程的补充。什么是⾯向切⾯编程呢? 切⾯就是指某⼀类特定问题, 所以AOP也可以理解为⾯向特定⽅法编程. 什么是⾯向特定⽅法编程呢? ⽐如上个章节学习的"登录校验", 就是⼀类特定问题. 登录校验拦截器, 就 是对"登录校验"这类问题的统⼀处理. 所以, 拦截器也是AOP的⼀种应⽤.AOP是⼀种思想, 拦截器是AOP 思想的⼀种实现. Spring框架实现了这种思想, 提供了拦截器技术的相关接⼝. 同样的, 统⼀数据返回格式和统⼀异常处理, 也是AOP思想的⼀种实现.简单来说: AOP是⼀种思想, 是对某⼀类事情的集中处理(用户登录校验,结果统一返回,异常统一处理)
拦截器作⽤的维度是URL(⼀次请求和响应), @ControllerAdvice 应⽤场景主要是全局异常处理 (配合⾃定义异常效果更佳), 数据绑定, 数据预处理.
AOP作⽤的维度更加细致(可以根据包、类、⽅法 名、参数等进⾏拦截), 能够实现更加复杂的业务逻辑.
一个项目需要开发很多的业务功能,有一些业务的执行效率是很低的,需要对接口进行优化。第一步就需要定位出执行耗时比较长的业务方法,然后针对该业务方法进行优化。如何定位呢?我们就需要统计当前项目中每一个业务方法的执行耗时,我们需要在每个业务方法运行前和运行后,记录下方法的开始时间和结束时间,两者之差就是这个方法的耗时。但是一个项目中有很多的方法和类,我们需要记录每个方法的执行时长,需要在每个方法中记录,这种执行效率是比较低的,耗时比较长,
AOP就可以做到在不改动这些原始⽅法的基础上, 针对特定的⽅法进⾏功能的增强.AOP的作⽤:在程序运⾏期间在不修改源代码的基础上对已有⽅法进⾏增强(⽆侵⼊性: 解耦)
🚩Spring AOP快速⼊⻔
需求: 统计图书系统各个接⼝⽅法的执⾏时间
🎓引入AOP依赖
在pom.xml⽂件中添加配置
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-aop</artifactId>
</dependency>
🎓编写AOP程序
记录Controller中每个⽅法的执⾏时间
@Slf4j
@Component
@Aspect
public class TimeRecordAspect {/*** 记录时间*/@Around("execution(* com.example.cl.controller.*.*(..))")//对哪些方法生效//@Around表示侵入程序的时机,表示环绕,目标函数执行前后都会执行public Object TimeRecord(ProceedingJoinPoint joinPoint) throws Throwable {//记录开始时间long start=System.currentTimeMillis();//目标函数执行前//执行目标方法Object proceed=joinPoint.proceed();//执行目标函数//记录结束时间long end=System.currentTimeMillis();//目标函数执行后//日志打印耗时时间log.info("耗时时间:"+(end-start)+" ms");return proceed;}//AOP这种写法作用的维度是 方法//拦截器 作用的维度是 url 接口
}
在进行用户登录,我们看到该方法耗时时间:234ms.
对程序进⾏简单的讲解:
- @Aspect: 标识这是⼀个切⾯类
- @Around: 环绕通知, 在⽬标⽅法的前后都会被执⾏. 后⾯的表达式表⽰对哪些⽅法进⾏增强.
- ProceedingJoinPoint.proceed() 让原始⽅法执⾏
我们通过AOP⼊⻔程序完成了业务接⼝执⾏耗时的统计.通过上⾯的程序, 我们也可以感受到 AOP⾯向切⾯编程的⼀些优势 :
- • 代码⽆侵⼊: 不修改原始的业务⽅法, 就可以对原始的业务⽅法进⾏了功能的增强或者是功能的改变
- • 减少了重复代码
- • 提⾼开发效率
- • 维护⽅便
🚩Spring AOP 详解
🎓Spring AOP核⼼概念
🎈切点(Pointcut)切点(Pointcut), 也称之为"切⼊点"Pointcut 的作⽤就是提供⼀组规则 (使⽤ AspectJ pointcut expression language 来描述), 告诉程序对 哪些⽅法来进⾏功能增强.@Around("execution(* com.example.cl.controller.*.*(..))") 这就是切点表达式🎈连接点(Join Point)满⾜切点表达式规则的⽅法, 就是连接点. 也就是可以被AOP控制的⽅法以⼊⻔程序举例, 所有 com.example.demo.controller 路径下的⽅法, 都是连接点UserController/BookController 中的⽅法都是连接点📝切点表达式和连接点的关系:连接点是满足切点表达式的元素,切点可以看作是保存了众多连接点的一个集合连接点——路径下里面的所有的方法切点表达式——路径🎈通知(Advice)通知就是具体要做的⼯作, 指哪些重复的逻辑,也就是共性功能(最终体现为⼀个⽅法)⽐如上述程序中记录业务⽅法的耗时时间, 就是通知
在AOP⾯向切⾯编程当中, 我们把这部分重复的代码逻辑抽取出来单独定义, 这部分代码就是通知的内容.🎈切⾯(Aspect)切⾯(Aspect) = 切点(Pointcut) + 通知(Advice)通过切⾯就能够描述当前AOP程序需要针对于哪些⽅法, 在什么时候执⾏什么样的操作切⾯既包含了通知逻辑的定义, 也包括了连接点的定义
切⾯所在的类, 我们⼀般称为切⾯类(被@Aspect注解标识的类)
🎓通知类型
Spring中AOP的通知类型有以下⼏种:
- @Around: 环绕通知, 此注解标注的通知⽅法在⽬标⽅法前, 后都被执⾏
- @Before: 前置通知, 此注解标注的通知⽅法在⽬标⽅法前被执⾏
- @After: 后置通知, 此注解标注的通知⽅法在⽬标⽅法后被执⾏, ⽆论是否有异常都会执⾏
- @AfterReturning: 返回后通知, 此注解标注的通知⽅法在⽬标⽅法后被执⾏, 有异常不会执⾏
- @AfterThrowing: 异常后通知, 此注解标注的通知⽅法发⽣异常后执⾏
@Aspect
@Slf4j
public class AspectDemo {/*** 前置通知*/@Before("execution(* com.cl.springaop.controller.*.*(..))")public void doBefore(){log.info("AspectDemo do before......");}//是否发生异常,都会执行(再执行方法前)优先级小于around/*** 后置通知*/@After("execution(* com.cl.springaop.controller.*.*(..))")public void doAfter(){log.info("AspectDemo do after......");}//是否发生异常,都会执行(再执行方法后) 优先级小于around/***添加环绕通知*///around必须有返回值,需要将目标返回值返回,否则会没有返回值@Around("execution(* com.cl.springaop.controller.*.*(..))")public Object doAround(ProceedingJoinPoint joinPoint) throws Throwable {log.info("AspectDemo do around before......");Object result=null;result=joinPoint.proceed();log.info("AspectDemo do around after.......");return result;}//发生异常就不会返回结果/*** 返回后通知*/@AfterReturning("execution(* com.cl.springaop.controller.*.*(..))")public void doAfterReturning(){log.info("AspectDemo do AfterReturning......");}//没有异常执行/*** 抛出异常后通知*/@AfterThrowing("execution(* com.cl.springaop.controller.*.*(..))")public void doAfterThrowing(){log.info("AspectDemo do AfterThrowing......");}//有异常执行
}
写一些测试程序:
@RestController
@Slf4j
@RequestMapping("/test")
public class TestController {@RequestMapping("/t1")public String t1(){log.info("执行t1方法");return "t1";}@RequestMapping("/t2")public String t2(){log.info("执行t2方法.......");int a=10/0;return "t2";}
}
1. 正常运⾏的情况
程序正常运⾏的情况下, @AfterThrowing 标识的通知⽅法不会执⾏从上图也可以看出来,@Around 标识的通知⽅法包含两部分, ⼀个"前置逻辑", ⼀个"后置逻辑". 其中"前置逻辑" 会先于 @Before 标识的通知⽅法执⾏, "后置逻辑" 会晚于 @After 标识的通知⽅法执⾏![]()
2.异常时的情况
程序发⽣异常的情况下:
- • @AfterReturning 标识的通知⽅法不会执⾏, @AfterThrowing 标识的通知⽅法执⾏了
- • @Around 环绕通知中原始⽅法调⽤时有异常,通知中的环绕后的代码逻辑也不会在执⾏了(因为原始⽅法调⽤出异常了)
注意事项:
- • @Around 环绕通知需要调⽤ ProceedingJoinPoint.proceed() 来让原始⽅法执⾏, 其他 通知不需要考虑⽬标⽅法执⾏.
- • @Around 环绕通知⽅法的返回值, 必须指定为Object, 来接收原始⽅法的返回值, 否则原始⽅法执⾏完毕, 是获取不到返回值的.
- • ⼀个切⾯类可以有多个切点.
🎓@PointCut
@Pointcut("execution(* com.cl.springaop.controller.*.*(..))")public void pt(){};
@Pointcut("execution(* com.cl.springaop.controller.*.*(..))")public void pt(){};/*** 前置通知*/@Before("pt()")public void doBefore(){log.info("AspectDemo do before......");}//是否发生异常,都会执行(再执行方法前)优先级小于around/*** 后置通知*/@After("pt()")public void doAfter(){log.info("AspectDemo do after......");}//是否发生异常,都会执行(再执行方法后) 优先级小于around
当切点定义使⽤private修饰时, 仅能在当前切⾯类中使⽤,当其他切⾯类也要使⽤当前切点定义时, 就需要把 private改为public.引⽤⽅式为: 全限定类名.⽅法名()![]()
@Component @Aspect @Slf4j public class AspectDemo2 {@Before("com.cl.springaop.aspect.AspectDemo.pt()")public void doBefore(){log.info(" AspectDemo2 do before......");} }
🎓切⾯优先级 @Order
为防⽌⼲扰, 我们把AspectDemo这个切⾯先去掉(把 @Component 注解去掉就可以 )


存在多个切⾯类时, 默认按照切⾯类的类名字⺟排序:
- • @Before 通知:字⺟排名靠前的先执⾏
- • @After 通知:字⺟排名靠前的后执⾏
但这种⽅式不⽅便管理, 我们的类名更多还是具备⼀定含义的.Spring 给我们提供了⼀个新的注解, 来控制这些切⾯通知的执⾏顺序: @Order
按照4,3,2的顺序在先执行。
@Order 注解标识的切⾯类, 执⾏顺序如下:
- • @Before 通知:数字越⼩先执⾏
- • @After 通知:数字越⼤先执⾏

🎓切点表达式
上⾯的代码中, 我们⼀直在使⽤切点表达式来描述切点. 下⾯我们来介绍⼀下切点表达式的语法.切点表达式常⻅有两种表达⽅式
- 1. execution(RR):根据⽅法的签名来匹配
- 2. @annotation(RR) :根据注解匹配
📝execution表达式
execution(< 访问修饰符 > < 返回类型 > < 包名 . 类名 . ⽅法 ( ⽅法参数 )> < 异常 >)

📝@annotation
实现步骤:
- 1. 编写⾃定义注解
- 2. 使⽤ @annotation 表达式来描述切点
- 3. 在连接点的⽅法上添加⾃定义注解
UserController
@Component
@RestController
@Slf4j
@RequestMapping("/user")
public class UserController {@RequestMapping("/u1")public String u1(){log.info("执行u1方法");return "u1";}//由于没有引入注解,所以没有@RequestMapping("/u2")public String u2(){log.info("执行u2方法");return "u2";}
}
TestController
@RestController
@Slf4j
@RequestMapping("/test")
public class TestController {@RequestMapping("/t1")public String t1(){log.info("执行t1方法");return "t1";}@RequestMapping("/t2")public String t2(){log.info("执行t2方法.......");int a=10/0;return "t2";}
}
🎈⾃定义注解 @MyAspect
@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface MyAspect {
}
🎈切⾯类
@Aspect
@Component
@Slf4j
public class MyAspectDemo {@Around("@annotation(com.cl.springaop.config.MyAspect)")public Object doAround(ProceedingJoinPoint joinPoint) {log.info("MyAspectDemo do around before.....");Object result= null;try {result = joinPoint.proceed();} catch (Throwable e) {log.error("发生异常,e:"+e);}log.info("MyAspectDemo do around after.....");return result;}
}
@Around("@annotation(org.springframework.web.bind.annotation.RequestMapping)")public Object doAround2(ProceedingJoinPoint joinPoint) {log.info("RequestMapping do around before.....");Object result= null;try {result = joinPoint.proceed();} catch (Throwable e) {log.error("发生异常,e:"+e);}log.info("RequestMapping do around after.....");return result;}
🎈添加⾃定义注解


对testController中的t2测试
没有打印结果。
🚩Spring AOP的实现⽅式(常⻅⾯试题)
- 1. 基于注解 @Aspect
- 2. 基于Spring API (通过xml配置的⽅式, ⾃从SpringBoot ⼴泛使⽤之后, 这种⽅法⼏乎看不到了)
- 3. 基于代理来实现(更加久远的⼀种实现⽅式, 写法笨重, 不建议使⽤)
背挺直一点,向前靠一点。
相关文章:

【JavaEE进阶】——Spring AOP
目录 🚩Spring AOP概述 🚩Spring AOP快速⼊⻔ 🎓引入AOP依赖 🎓编写AOP程序 🚩Spring AOP 详解 🎓Spring AOP核⼼概念 🎓通知类型 🎓PointCut 🎓切⾯优先级 Ord…...

Python - conda使用大全
如何使用Conda? 环境 创建环境 conda create -n spider_env python3.10.11查看环境 conda env listconda info -e激活环境 conda activate spider_env退出环境 conda deactivate删除环境 conda env remove -n spider_env包 导出包 说明:导出当前虚拟…...

ASPICE在汽车软件开发中的作用
ASPICE是一个专门为汽车软件开发过程而设计的评估和改进框架。它基于ISO/IEC 15504标准,为汽车供应商提供了一个评估和改进其软件开发流程的方法。ASPICE的目标是确保软件开发过程的一致性和可预测性,从而提高软件的质量和可靠性。 ASPICE的实施对汽车软…...

亚马逊云科技 re:Inforce 2024中国站大会
亚马逊云科技 re:Inforce 2024中国站大会 - 生成式AI时代的全面安全,将于7月25日本周四在北京富力万丽酒店揭幕...

Lottie:动态动画的魔法棒
文章目录 引言官网链接Lottie 的原理基础使用1. 导出动画2. 引入 Lottie 库3. 加载和播放动画 高级使用1. 动画控制2. 交互性3. 自定义动画例子:交互式按钮动画 优缺点优点缺点 结语 引言 Lottie 是 Airbnb 开源的一个动画库,它允许设计师在 Adobe Afte…...

IPython使用技巧整理
IPython 是一个增强的 Python 交互式 shell,它提供了许多便利的功能,比如自动补全、魔术命令、对象内省等。以下是 IPython 的一些使用技巧和示例,结合您提供的列表数据,我将给出一些相关的使用示例。 1. 自动补全(Tab…...

C#数组复习
一、一维数组 using System.Collections; using System.Collections.Generic; using UnityEngine;public class ShuZu : MonoBehaviour {#region 知识点一 基本概念//数组是存储一组相同类型数据的集合//数组分为 一维、二维、交错数组//一般情况 一维数组 就简称为数组#en…...

无人机之在农业上的用途
随着无人机技术的发展,农业现代化也迎来了崭新局面,田间随处可见无人机矫健的身影。当农业遇上科技,变革正悄然进行。农业无人机主要应用于农业、种植业、林业等行业。在使用过程中,其功能和作用并不单一,一般用于种植…...

opengaussdb在oepnEuler上安装
安装前提: 软件环境:openEuler 20.03LTS 个人开发者最低配置2核4G,推荐配置4核8G 数据库版本:openGauss-5.0.2-openEuler-64bit-all.tar.gz 数据库下载地址: https://docs-opengauss.osinfra.cn/zh/docs/5.0.0/docs/In…...

一些和颜色相关网站
1.中国传统色 2.网页颜色选择器 3.渐变色网站 4.多风格色卡生成 5.波浪生成 6.半透明磨砂框 7.色卡组合...

Linux系统编程-文件系统
目录 什么是Linux文件系统 文件系统的职责 存储介质抽象 inode:文件系统的核心 文件分配策略 目录结构 文件系统布局 日志和恢复机制 目录权限 粘滞位(t位): 硬链接和符号链接 硬链接的特点: 创建硬链接: 符号链接的…...

【解决】ubuntu20.04 root用户无法SSH登陆问题
Ubuntu root用户无法登录的问题通常可以通过修改SSH配置文件和系统登录配置来解决。 修改SSH配置文件 sudo vim /etc/ssh/sshd_config 找到 PermitRootLogin 设置,并将其值更改为 yes 以允许root用户通过SSH登录 保存并关闭文件之后,需要重启SSH服务…...

(前缀和) LeetCode 238. 除自身以外数组的乘积
一. 题目描述 原题链接 给你一个整数数组 nums,返回 数组 answer ,其中 answer[i] 等于 nums 中除 nums[i] 之外其余各元素的乘积 。 题目数据 保证 数组 nums之中任意元素的全部前缀元素和后缀的乘积都在 32 位 整数范围内。 请 不要使用除法&…...

【JVM基础05】——组成-能不能解释一下方法区?
目录 1- 引言:方法区概述1-1 方法区是什么?(What)1-2 为什么用方法区?方法区的作用 (Why) 2- ⭐核心:详解方法区(How)2-1 能不能解释一下方法区?2-2 元空间内存溢出问题2-3 什么是常量池?2-4 运行时常量池 …...

前端:Vue学习-3
前端:Vue学习-3 1. 自定义指令2. 插槽2.1 插槽 - 后备内容(默认值)2.2 插槽 - 具名插槽2.3 插槽 - 作用域插槽 3. Vue - 路由3.1 路由模块封装3.2 声明式导航 router-link 高亮3.3 自定义匹配的类名3.4 声明式导肮 - 跳转传参3.5 Vue路由 - 重…...

npm 安装报错(已解决)+ 运行 “wue-cli-service”不是内部或外部命令,也不是可运行的程序(已解决)
首先先说一下我这个项目是3年前的一个项目了,中间也是经过了多个人的修改惨咋了布置多少个人的思想,这这道我手里直接npm都安装不上,在网上也查询了多种方法,终于是找到问题所在了 问题1: 先是npm i 报错在下面图片&…...

江苏科技大学24计算机考研数据速览,有专硕复试线大幅下降67分!
江苏科技大学(Jiangsu University of Science and Technology),坐落在江苏省镇江市,是江苏省重点建设高校,江苏省人民政府与中国船舶集团有限公司共建高校,国家国防科技工业局与江苏省人民政府共建高校 &am…...

20分钟上手新版Skywalking 9.x APM监控系统
Skywalking https://skywalking.apache.org/ Skywalking是专为微服务、云原生和基于容器的(Kubernetes)架构设计的分布式系统性能监控工具。 Skywalking关键特性 ● 分布式跟踪 ○ 端到端分布式跟踪。服务拓扑分析、以服务为中心的可观察性和API仪表板。…...

【07】LLaMA-Factory微调大模型——微调模型导出与微调参数分析
上文介绍了如何对微调后的模型进行使用与简单评估。本文将介绍对微调后的模型进行导出的过程。 一、llama-3微调后的模型导出 首先进入虚拟环境,打开LLaMA-Factory的webui页面 conda activate GLM cd LLaMA-Factory llamafactory-cli webui 之后,选择…...

动态路由协议 —— EIGRP 与 OSPF 的区别
EIGRP(增强内部网关路由协议)和 OSPF(开放式最短路径优先)是两种最常见的动态路由协议,主要是用来指定路由器或交换机之间如何通信。将其应用于不同的情况下,可提高速率、延迟等方面的性能。那么它们之间到…...

【中项】系统集成项目管理工程师-第5章 软件工程-5.1软件工程定义与5.2软件需求
前言:系统集成项目管理工程师专业,现分享一些教材知识点。觉得文章还不错的喜欢点赞收藏的同时帮忙点点关注。 软考同样是国家人社部和工信部组织的国家级考试,全称为“全国计算机与软件专业技术资格(水平)考试”&…...

HarmonyOS应用开发者高级认证,Next版本发布后最新题库 - 多选题序号1
基础认证题库请移步:HarmonyOS应用开发者基础认证题库 注:有读者反馈,题库的代码块比较多,打开文章时会卡死。所以笔者将题库拆分,单选题20个为一组,多选题10个为一组,题库目录如下,…...

Windows11(24H2)LTSC长期版下载!提前曝光Build26100?
系统;windows11 文章目录 前言一、LTSC是什么?二、 Windows 11 Vision 24H2 LTSC 的版本号为 Build 26100,镜像中提供以下三个 SKU:总结 前言 好的系统也能给你带来不一样的效果。 一、LTSC是什么? & & L…...

【北京迅为】《i.MX8MM嵌入式Linux开发指南》-第三篇 嵌入式Linux驱动开发篇-第四十三章 驱动模块传参
i.MX8MM处理器采用了先进的14LPCFinFET工艺,提供更快的速度和更高的电源效率;四核Cortex-A53,单核Cortex-M4,多达五个内核 ,主频高达1.8GHz,2G DDR4内存、8G EMMC存储。千兆工业级以太网、MIPI-DSI、USB HOST、WIFI/BT…...

uniapp 小程序 支付逻辑处理
uniapp 小程序 支付逻辑处理 上代码如果你不需要支付宝适配,可以删除掉支付宝的条件判断代码 <button class"subBtn" :disabled"submiting" click"goPay">去支付</button>// 以下代码你需要改的地方// 1. order/app/v1…...

scikit-learn库学习之make_regression函数
scikit-learn库学习之make_regression函数 一、简介 make_regression是scikit-learn库中用于生成回归问题数据集的函数。它主要用于创建合成的回归数据集,以便在算法的开发和测试中使用。 二、语法和参数 sklearn.datasets.make_regression(n_samples100, n_feat…...

经典文献阅读之--World Models for Autonomous Driving(自动驾驶的世界模型:综述)
Tip: 如果你在进行深度学习、自动驾驶、模型推理、微调或AI绘画出图等任务,并且需要GPU资源,可以考虑使用UCloud云计算旗下的Compshare的GPU算力云平台。他们提供高性价比的4090 GPU,按时收费每卡2.6元,月卡只需要1.7元每小时&…...

孙健提到的实验室的研究方向之一是什么?()
孙健提到的实验室的研究方向之一是什么?() 点击查看答案 A.虚拟现实B.环境感知和理解 C.智能体博弈D.所有选项都正确 图灵奖是在哪一年设立的?() A.1962B.1966 C.1976D.1986 孙健代表的实验室的前身主要研究什么?&…...

初级java每日一道面试题-2024年7月23日-Iterator和ListIterator有什么区别?
面试官: Iterator和ListIterator有什么区别? 我回答: Iterator和ListIterator都是Java集合框架中用于遍历集合元素的接口,但它们之间存在一些关键的区别,主要体现在功能和使用场景上。下面我将详细解释这两种迭代器的不同之处: 1. Iterat…...

2024-07-23 Unity AI行为树2 —— 项目介绍
文章目录 1 项目介绍2 AI 代码介绍2.1 BTBaseNode / BTControlNode2.2 动作/条件节点2.3 选择 / 顺序节点 3 怪物实现4 其他功能5 UML 类图 项目借鉴 B 站唐老狮 2023年直播内容。 点击前往唐老狮 B 站主页。 1 项目介绍 本项目使用 Unity 2022.3.32f1c1,实现基…...