黑马Spring学习笔记(四)——面向切面编程AOP

news2025/1/14 14:32:16

目录

一、AOP简介

二、AOP核心概念

三、AOP入门案例

四、AOP配置管理

4.1  AOP切入点表达式

4.1.1  语法格式

4.2.2  通配符

4.2.3  书写技巧

4.2  AOP通知类型

4.2.1 前置、后置、返回后、抛出异常后获取参数

4.2.2  环绕通知


一、AOP简介

        AOP(Aspect Oriented Programming)面向切面编程,一种编程范式,也是一种编程思想,编程思想主要的内容就是指导程序员该如何编写程序。

        AOP的作用是:在不惊动原始设计的基础上为其进行功能增强。


        下面我们来看一个例子:

        最主要的类BookDaoImpl内容如下:

@Repository
    public class BookDaoImpl implements BookDao {
        public void save() {
        //记录程序当前执行执行(开始时间)
        Long startTime = System.currentTimeMillis();
        //业务执行万次
        for (int i = 0;i<10000;i++) {
            System.out.println("book dao save ...");
        }
        //记录程序当前执行时间(结束时间)
        Long endTime = System.currentTimeMillis();
         //计算时间差
         Long totalTime = endTime-startTime;
        //输出信息
         System.out.println("执行万次消耗时间:" + totalTime + "ms");
     }

     public void update(){
         System.out.println("book dao update ...");
     }

     public void delete(){
        System.out.println("book dao delete ...");
     }

    public void select(){
         System.out.println("book dao select ...");
     }
 }

        不难看出,这个类中有四个方法,其中save()方法有计算万次执行消耗的时间。

        但是,当我们从容器中获取bookDao对象后,分别对其执行save(),delete(),update(),select(),会有如下的打印结果:

 为什么delete()和update()方法也执行了10000次且计算了执行时间呢?

        这就是SpringAOP,在不惊动(改动)原有设计(代码)的前提下,想给谁添加功能就给谁添加。

二、AOP核心概念

        那Spring到底是如何实现AOP的呢?

  1.  连接点与切入点

    连接点:执行的方法

    切入点:需要被增强的方法

    连接点: 程序执行过程中的任意位置,可以为方法,抛出异常,也可以为设置变量。

    切入点: 一个切入点可以描述一个具体的方法,也可以匹配多个方法,一个切入点可以只匹配一个update方法,也可以匹配某个包下面所有的查询方法。

    【连接点范围 > 切入点范围,切入点一定是连接点,反之未必】

  2. 通知与通知类

    通知:存放共性功能的方法

    通知类:定义通知的类

    通知: 在切入点处执行的操作,也就是共性功能。

    通知类: 通知是一个方法,方法不能独立存在,需要被写在一个类中,这个类我们也给起了个名字叫通知类

  3. 切面

    切面:通知与切入点之间的关系描述

    通知是要增强的内容,会有多个;切入点是需要被增强的方法,也会有多个,哪个切入点需要添加哪个通知,就需要提前将它们之间的关系描述清楚,那么对于通知和切入点之间的关系描述,我们称之为切面

三、AOP入门案例

        我们使用注解来完成AOP的开发。

        案例为:使用SpringAOP的注解方式完成在方法执行前打印出当前系统时间。

3.1 思路分析:

1. 导入坐标 (pom.xml)
2. 制作连接点 ( 原始操作, Dao 接口与实现类 )
3. 制作共性功能 ( 通知类与通知 )
4. 定义切入点
5. 绑定切入点与通知关系 ( 切面 )

3.2 实现步骤:

环境准备
  • 创建一个Maven项目
  • pom.xml 添加 Spring 依赖
    <dependencies>
         <dependency>
             <groupId>org.springframework</groupId>
             <artifactId>spring-context</artifactId>
             <version>5.3.20</version>
        </dependency>
    </dependencies>
  • 添加 BookDao BookDaoImpl
    public interface BookDao {
        public void save();
    
        public void update();
    }
    
    @Repository
    public class BookDaoImpl implements BookDao {
        public void save() {
            System.out.println(System.currentTimeMillis());
            System.out.println("book dao save ...");
        }
    
        public void update() {
            System.out.println("book dao update ...");
        }
    }
  • 创建Spring的配置类
    @Configuration
    @ComponentScan("com.itheima")
    public class SpringConfig {
    }

    项目结构如下:

AOP实现步骤
  1. 添加依赖
    <dependency>
        <groupId>org.aspectj</groupId>
        <artifactId>aspectjweaver</artifactId>
        <version>1.9.4</version>
    </dependency>

    因为spring-context中已经导入了spring-aop ,所以不需要再单独导入spring-aop

  2. 定义接口与实现类
    BookDaoImpl
    已经准备好,不需要做任何修改
  3. 定义通知类和通知
    public class MyAdvice {
        public void method(){
            System.out.println(System.currentTimeMillis());
        }
    }
    因类名和方法名没有要求,可以任意
  4. 定义切入点
    BookDaoImpl
    中有两个方法,分别是 save update ,我们要增强的是 update 方法,该如何定义呢 ?
    public class MyAdvice {
        @Pointcut("execution(void com.itheima.dao.BookDao.update())")
        private void pt() {
        }
    
        public void method() {
            System.out.println(System.currentTimeMillis());
        }
    }
    切入点定义依托一个不具有实际意义的方法进行,即无参数、无返回值、方法体无实际逻辑
  5. 制作切面
    public class MyAdvice {
        @Pointcut("execution(void com.itheima.dao.BookDao.update())")
        private void pt() {
        }
    
        @Before("pt()")
        public void method() {
            System.out.println(System.currentTimeMillis());
        }
    }
    绑定切入点与通知关系,并指定通知添加到原始连接点的具体执行 位置
  6. 将通知类配给容器并标识其为切面类
    @Component
    @Aspect
    public class MyAdvice {
        @Pointcut("execution(void com.itheima.dao.BookDao.update())")
        private void pt() {
        }
    
        @Before("pt()")
        public void method() {
            System.out.println(System.currentTimeMillis());
        }
    }
  7. 开启注解格式AOP功能
    @Configuration
    @ComponentScan("com.itheima")
    @EnableAspectJAutoProxy
    public class SpringConfig {
    }
  8. 运行程序
    public class App {
        public static void main(String[] args) {
            ApplicationContext ctx = new
                    AnnotationConfigApplicationContext(SpringConfig.class);
            BookDao bookDao = ctx.getBean(BookDao.class);
            bookDao.update();
        }
    }

四、AOP配置管理

4.1  AOP切入点表达式

4.1.1  语法格式

execution(public User com.itheima.service.UserService.findById(int))


参数:

execution :动作关键字,描述切入点的行为动作,例如 execution 表示执行到指定切入点
public: 访问修饰符 , 还可以是 public private 等,可以省略
User :返回值,写返回值类型
com.itheima.service :包名,多级包使用点连接
UserService: / 接口名称
findById :方法名
int: 参数,直接写参数的类型,多个类型用逗号隔开
异常名:方法定义中抛出指定异常,可以省略
        切入点表达式就是要找到需要增强的方法,所以它就是对一个具体方法的描述,但是方法的定义会有很多,所以如果每一个方法对应一个切入点表达式,想想这块就会觉得将来编写起来会比较麻烦,就需要用到下面所学习的通配符。

4.2.2  通配符

  1.   *  单个独立的任意符号,可以独立出现,也可以作为前缀或者后缀的匹配符出现
    execution(public * com.itheima.*.UserService.find*(*))
  2.   ..   :多个连续的任意符号,可以独立出现,常用于简化包名与参数的书写
    execution(public User com..UserService.findById(..))
  3.   +   :专用于匹配子类类型
    execution(* *..*Service+.*(..))
    这个使用率较低,描述子类的,JavaEE开发,继承机会就一次,使用都很慎重,所以很少用它。*Service+,表示所有以Service结尾的接口的子类

例如:

execution(void com..*())
返回值为voidcom包下的任意包任意类任意方法,能匹配,*代表的是方法
execution(* com.itheima.*.*Service.find*(..))
将项目中所有业务层方法的以find开头的方法匹配
execution(* com.itheima.*.*Service.save*(..))
将项目中所有业务层方法的以save开头的方法匹配
       

4.2.3  书写技巧

  • 所有代码按照标准规范开发,否则以下技巧全部失效
  • 描述切入点通常描述接口,而不描述实现类,如果描述到实现类,就出现紧耦合了
  • 访问控制修饰符针对接口开发均采用public描述(可省略访问控制修饰符描述
  • 返回值类型对于增删改类使用精准类型加速匹配,对于查询类使用*通配快速描述
  • 包名书写尽量不使用..匹配,效率过低,常用*做单个包描述匹配,或精准匹配
  • 接口名 / 类名 书写名称与模块相关的 采用 * 匹配 ,例如 UserService 书写成 *Service ,绑定业务
    层接口名
  • 方法名 书写以 动词 进行 精准匹配 ,名词采用 匹配,例如 getById 书写成 getBy ,selectAll 书写成
    selectAll
  • 参数规则较为复杂,根据业务方法灵活调整
  • 通常不使用异常作为匹配规则

4.2  AOP通知类型

        AOP提供了5种通知类型:

  • 前置通知
  • 后置通知
  • 环绕通知(重点)
  • 返回后通知(了解)
  • 抛出异常后通知(了解)

        我们来看一张图:

4.2.1 前置、后置、返回后、抛出异常后获取参数

目前有执行方法(连接点、切入点)findName()如下:

@Repository
public class BookDaoImpl implements BookDao {

     public String findName(int id) {
         return "itcast";
     }
}

共性方法(通知类)如下:

@Component
@Aspect
public class MyAdvice {
    @Pointcut("execution(* com.itheima.dao.BookDao.findName(..))")
    private void pt() {
    }

    @Before("pt()")
    public void before() {
        System.out.println("before advice ...");
    }

    @After("pt()")
    public void after() {
        System.out.println("after advice ...");
    }
}

那么当对findName()进行增强的时候,通知类中的共性方法如何才能获取findName中的参数id呢?


  • 使用JoinPoint

通知作如下修改:

@Before("pt()")
public void before(JoinPoint jp)
    Object[] args = jp.getArgs();
    System.out.println(Arrays.toString(args));

    System.out.println("before advice ..." );
}

我们在执行类执行findName(100)时,输出如下:

可见,共性方法成功得到了执行方法中的id参数,并且以数组的形式进行了输出(因为方法中的形参可能有多个)

如果执行方法为findName(id, password),在执行类执行findName(100,itheima),那么输出为:

4.2.2  环绕通知

1. 环绕通知不能像前置和后置通知一样,简单地写成

@Around("pt()")
public void around(){
    System.out.println("around before advice ...");
    System.out.println("around after advice ...");
}
【❌❌❌❌❌】

而应该写成

@Around("pt()")
public void around(ProceedingJoinPoint pjp) throws Throwable{
    System.out.println("around before advice ...");
    //表示对原始操作的调用
    pjp.proceed();
    System.out.println("around after advice ...");
}

通过proceed()函数可以区分环绕前面和后面的通知

2. 如果我们使用环绕通知的话,要根据原始方法的返回值来设置环绕通知的返回值

代码示例:

@Around("pt2()")
public Object aroundSelect(ProceedingJoinPoint pjp) throws Throwable {
    System.out.println("around before advice ...");
    //表示对原始操作的调用
    Object ret = pjp.proceed();
    System.out.println("around after advice ...");
    return ret;
}
为什么返回的是 Object 而不是 int 的主要原因是 Object 类型更通用。

3. 环绕通知获取参数

与非环绕通知的JoinPoint类似,环绕通知使用ProceedingJoinPoint,示例:

@Around("pt()")
public Object around(ProceedingJoinPoint pjp)throws Throwable {
    Object[] args = pjp.getArgs();
    System.out.println(Arrays.toString(args));

    Object ret = pjp.proceed();
    return ret;
}

同时,我们还可修改原始方法的参数,通过 args[0] = 666,我们就可以在环绕通知中对原始方法的参数进行拦截过滤,避免由于参数的问题导致程序无法正确运行,保证代码的健壮性。

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

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

相关文章

【IoT】创业成功不可或缺的两个因素:能力和趋势

今天就来谈谈能力和趋势究竟哪个更重要的问题。 在谈成功的十大要素时&#xff0c;我曾经讲到&#xff1a; 一命、二运、三风水&#xff0c;这三个要素几乎不涉及任何个人的努力。 而趋势跟这三个要素又是息息相关的&#xff0c;这也类似雷军所说的飞猪理论。 只要风足够大&…

Centos7(阿里云)_安装Mysql8.0

1.安装MySQL 新人可以试用一个月的阿里云&#xff0c;centos7的 一开始可能确实会自带mariadb&#xff0c;所以可以在网上随便找个教程开始尝试安装MySQL&#xff0c;当然大概率出错&#xff0c;然后此时你的rpm下面已经有了一个版本的mysql安装包。 以我为例&#xff0c;随便…

Windows下jdk安装与卸载-超详细的图文教程

jdk安装 下载jdk 由于现在主流就是jdk1.8&#xff0c;所以这里就下载jdk1.8进行演示。官方下载地址&#xff1a;https://www.oracle.com/java/technologies/downloads/#java8-windows。 官方下载需要注册oracle账号&#xff0c;国内下载有可能速度慢&#xff0c;若不想注册账…

【Linux】P5 实用快捷键 以及 下载安装指令

Linux实用快捷键与下载安装实用快捷键强制停止退出账户 / 环境查看历史命令光标移动快捷键清屏下载安装命令CentOS - yumUbunto - apt实用快捷键 强制停止 CTRL C当程序陷入死循环或者执行错误&#xff0c;可通过该命令强制停止 当程序输入很长但是发现错误&#xff0c;不想…

PDF 解析格式化输出 API 数据接口

PDF 解析格式化输出 API 数据接口 支持输出 TEXT HTML XML TAG&#xff0c;多种格式输出&#xff0c;超精准识别率。 1. 产品功能 通用的识别接口&#xff0c; 支持标准 PDF 文件解析&#xff1b;多种格式输出&#xff0c;支持 TEXT HTML XML TAG&#xff1b;HTML 包含完美排…

蒙特卡洛树搜索(MTCS)

一、目标 一种启发式的搜索算法&#xff0c;在搜索空间巨大的场景下比较有效 算法完成后得到一棵树&#xff0c;这棵树可以实现&#xff1a;给定一个游戏状态&#xff0c;直接选择最佳的下一步 二、算法四阶段 1、选择&#xff08;Selection&#xff09; 父节点选择UCB值最…

【ONE·C || 文件操作】

总言 C语言&#xff1a;文件操作。    文章目录总言1、文件是什么&#xff1f;为什么需要文件&#xff1f;1.1、为什么需要文件&#xff1f;1.2、文件是什么&#xff1f;2、文件的打开与关闭2.1、文件指针2.2、文件打开和关闭&#xff1a;fopen、fclose2.3、文件使用方式3、文…

windows下go安装并使用protobuf

go使用protobuf的过程以及可能出现的问题1. 下载proto windows版本并安装2. 安装protoc-gen-go3. proto文件转go文件报错protoc-gen-go: unable to determine go import path for "xxxx.proto"4. 生成的go文件中google.golang.org/protobuf/reflect/protoreflect依赖…

Zookeeper3.5.7版本——客户端命令行操作(znode 节点数据信息)

目录一、命令行语法二、znode 节点数据信息2.1、查看当前znode中所包含的内容2.2、查看当前节点详细数据2.3、节点详细数据解释一、命令行语法 命令行语法列表 命令基本语法功能描述help显示所有操作命令ls path使用 ls 命令来查看当前 znode 的子节点 [可监听]-w 监听子节点变…

粒子群优化最小二乘支持向量机SVM回归分析,pso-lssvm回归预测

目录 支持向量机SVM的详细原理 SVM的定义 SVM理论 SVM应用实例,粒子群优化最小二乘支持向量机SVM回归分析 代码 结果分析 展望 支持向量机SVM的详细原理 SVM的定义 支持向量机(support vector machines, SVM)是一种二分类模型,它的基本模型是定义在特征空间上的间隔最大…

视觉SLAM十四讲ch6 非线性优化笔记

视觉SLAM十四讲ch6 非线性优化笔记本讲目标上讲回顾状态估计问题非线性最小二乘Gauss-Newton&#xff1a;高斯牛顿Levenburg-Marquadt&#xff1a;列文伯格-马夸尔特小结实践&#xff1a;CERES实践&#xff1a;G2O本讲目标 理解最小二乘法的含义和处理方式。 理解Gauss-Newton…

CopyOnWriteArrayList 源码解读

一、CopyOnWriteArrayList 源码解读 在 JUC 中&#xff0c;对于 ArrayList 的线程安全用法&#xff0c;比较推崇于使用 CopyOnWriteArrayList &#xff0c;那 CopyOnWriteArrayList是怎么解决线程安全问题的呢&#xff0c;本文带领大家一起解读下 CopyOnWriteArrayList 的源码…

Day908.joinsnljdist和group问题和备库自增主键问题 -MySQL实战

join&snlj&dist和group问题和备库自增主键问题 Hi&#xff0c;我是阿昌&#xff0c;今天学习记录的是关于join&snlj&dist和group问题和备库自增主键问题的内容。 一、join 的写法 join 语句怎么优化&#xff1f;中&#xff0c;在介绍 join 执行顺序的时候&am…

基础复习第二十三天  File类与IO流的使用

java.io.File类:文件或文件目录路径的抽象表现形式&#xff0c;与平台无关。概述File能新建、删除、重命名文件或目录&#xff0c;但File不能访问文件内容本身。如果需要访问文件内容本身&#xff0c;则需要使用输入/输出流。想要在Java程序中表示一个真实存在的文件或目录&…

【Java并发编程】CountDownLatch

CountDownLatch是JUC提供的解决方案 CountDownLatch 可以保证一组子线程全部执行完牛后再进行主线程的执行操作。例如&#xff0c;主线程启动前&#xff0c;可能需要启动并执行若干子线程&#xff0c;这时就可以通过 CountDownLatch 来进行控制。 CountDownLatch是通过一个线程…

6. unity之脚本

1. 说明 当整个游戏运行起来之后&#xff0c;我们无法再借助鼠标来控制物体&#xff0c;此时可以使用脚本来更改物体的各种姿态&#xff0c;驱动游戏的整体运动逻辑。 2. 脚本添加 首先在Assets目录中&#xff0c;新创建一个Scripts文件夹&#xff0c;在该文件内右键鼠标选择…

opengl、opengl es、webgl介绍与opengl开发入门

1、OpenGL OpenGL&#xff08;英语&#xff1a;Open Graphics Library&#xff0c;译名&#xff1a;开放图形库或者“开放式图形库”&#xff09;常用于CAD、虚拟现实、科学可视化程序和电子游戏开发。OpenGL的高效实现&#xff08;利用了图形加速硬件&#xff09;存在于Windo…

HCIP-5OSPF区域类型学习笔记

1、OSPF区域类型 OSPF提出了区域的概念&#xff08;AREA&#xff09;&#xff0c;区域是将所有运行OSPF 的路由器人为的分成不同的组&#xff0c;以区域ID来标示。在区域内路由计算的方法不变&#xff0c;由于划分区域之后&#xff0c;每个区域内的路由器不会很多&#xff0c;…

【Python从入门到进阶】10、流程控制语句-循环语句(for-while)

接上篇《9、流程控制语句-条件语句(if-else)》 上一篇我们学习了Python的控制流语句的概念&#xff0c;以及其中的条件语句(if/else)&#xff0c;本篇我们来学习控制流语句中的循环语句(for/while)。 一、Python中的循环 Python的循环结构就是让程序“杀个回马枪”&#xff0…

修复电子管

年前在咸鱼捡漏买到了10根1G4G电子管&#xff0c;这是一种直热三极管&#xff0c;非常的少见。买回来的时候所有的灯丝都是通的&#xff0c;卖家说都是新的&#xff0c;库存货&#xff0c;但是外观实在是太糟糕了&#xff0c;看着就像被埋在垃圾场埋了几十年的那种&#xff0c;…