【Spring】19 AOP介绍及实例详解

news2024/11/14 15:15:41

文章目录

    • 1. 定义
      • 1)什么意思呢?
      • 2)如何解决呢?
    • 2. 基本概念
      • 1)切面(Aspect)
      • 2)切点(Pointcut)
      • 3)通知(Advice)
      • 4)连接点(Join Point)
      • 5)织入(Weaving)
    • 3. 例子
      • 1)添加依赖
      • 2)定义一个测试类
      • 3)创建切面
      • 4)创建启动类
      • 5)运行应用程序
    • 4. 将 [基本概念] 和 [例子] 结合解释说明
    • 总结

Spring 框架提供了强大的面向切面编程(AOP)支持,使得我们能够更方便的实现横切关注点的功能,比如日志记录、事务管理等。本文将深入探讨 Spring AOP 的概念、原理以及如何在应用程序中应用 AOP。

1. 定义

AOP 是一种编程范式,它允许将横切关注点与业务逻辑分离。横切关注点是那些跨足多个模块的关注点,比如日志、事务、方法执行时间等。通过 AOP 我们可以在不修改原始业务逻辑代码的情况下,将这些关注点模块化。

1)什么意思呢?

概念往往都是让人读不懂的,接下来让我们来举例说明一下:

首先看看我们的需求是什么呢?我们在实际开发过程中会写了无数个方法,想在每个方法开头和结束打印一下日志,例如某某方法开始了和某某方法结束了。

如下图这样:

在这里插入图片描述

是不是可以很清晰的发现一个问题,有无数个方法我们就需要在无数个方法里面分别加上这两个代码。。。

2)如何解决呢?

使用 AOP 面向切面编程,统一来一下。这就好像我们想把一些纸剪成两半似的,一张一张的剪就很慢(和上面现象一样),我们可以给它们摞起来然后用剪刀一刀下去,咔!全部剪开(AOP面向切面编程)。

就变成了下图这样:

在这里插入图片描述

是不是突然发现爽歪歪啦!

2. 基本概念

1)切面(Aspect)

切面是一种模块化的单元,它包含了横切关注点的实现,在 Spring AOP 中,切面是由切点和通知组成的。

2)切点(Pointcut)

切点定义了在何处应用切面的规则,它是一个表达式,用于匹配连接点(在应用中执行的特定代码位置),切点决定了切面在哪里生效。

3)通知(Advice)

通知是切面的具体实现,它定义了在何时、何地、以及如何应用切面的逻辑。常见的通知类型包括前置通知、后置通知、异常通知、最终通知和环绕通知。

4)连接点(Join Point)

连接点是在应用中执行的特定代码位置,如方法调用、方法执行、异常抛出等,它是切点的实际执行实例。

5)织入(Weaving)

织入是将切面应用到目标对象并创建代理对象的过程。织入可以在编译时、类加载时、运行时进行。

3. 例子

1)添加依赖

pom.xml 中添加 spring-aopaspectjweaver 两个依赖(这里只写了本次新添加的两个)

        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-aop</artifactId>
        </dependency>
		<dependency>
			<groupId>org.aspectj</groupId>
			<artifactId>aspectjweaver</artifactId>
			<version>1.9.20.1</version>
		</dependency>

2)定义一个测试类

创建一个简单的业务类,例如 DemoController ,其中包含一个待切入的方法

package com.cheney.koala.controller;

import org.springframework.web.bind.annotation.RestController;

@RestController
public class DemoController {
    
    public void print(){
        System.out.println("Hello");
    }
}

3)创建切面

创建一个切面类,例如 LoggingAspect ,用于定义在方法执行前后打印日志的通知

package com.cheney.koala.aspect;

import org.aspectj.lang.JoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import org.aspectj.lang.annotation.Pointcut;
import org.springframework.stereotype.Component;

@Aspect
@Component
public class LoggingAspect {

    @Pointcut("execution( * com.cheney.koala.controller.*.*(..) )")
    public void demoActionExecution() {
    }

    @Before("demoActionExecution()")
    public void logBefore(JoinPoint joinPoint) {
        System.out.println("方法 " + joinPoint.getSignature().getName() + " 开始啦");
    }

    @After("demoActionExecution()")
    public void logAfter(JoinPoint joinPoint) {
        System.out.println("方法 " + joinPoint.getSignature().getName() + " 结束啦");
    }
}

在这个切面中,我们使用 @Aspect 注解标识这是一个切面类。

@Pointcut("execution( * com.cheney.koala.controller.*.*(..) )") 这句代码是一个 Spring AOP 中的切点表达式,用于定义在哪些方法上应用切面。

  • @Pointcut:这是一个注解,用于定义切点,即在哪些方法上应用切面
  • execution:这是切点表达式的一部分,表示匹配方法的执行
  • *:这是通配符,表示匹配任意返回类型的方法
  • com.cheney.koala.controller:这是包名,表示匹配该包下的所有类
  • *.*(..):这也是通配符,表示匹配该包下所有类的所有方法

因此,整个表达式的意思就是匹配 com.cheney.koala.controller 包下所有类的所有方法,无论返回类型和参数类型是什么,都是我想切的对象(也即前后加入方法开始和方法结束的日志)

@Before@After 注解的通知方法中,使用 JoinPoint 获取连接点信息,包括方法签名等

4)创建启动类

KoalaApplication.java 中,添加了默认启动后调用 DemoController 里的 print 方法,仅仅是为了测试方便,没有别的特殊用意。

package com.cheney.koala;

import com.cheney.koala.controller.DemoController;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.context.ConfigurableApplicationContext;

@SpringBootApplication
public class KoalaApplication {
    public static void main(String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(KoalaApplication.class, args);

        DemoController demo = context.getBean(DemoController.class);
        demo.print();
    }
}

5)运行应用程序

运行应用程序,你将看到如下输出

在这里插入图片描述

方法 print 开始啦
Hello
方法 print 结束啦

这表明我们成功地通过 AOP 切面在方法执行前后分别打印了日志

4. 将 [基本概念] 和 [例子] 结合解释说明

在这里插入图片描述

  • **切面 **

    使用 @Aspect 注解标识这是一个切面类。在类中使用 @Pointcut 注解定义了一个切点 demoActionExecution,用于匹配我想切的对象

  • 切点

    execution( * com.cheney.koala.controller.*.*(..) ) 表示 com.cheney.koala.controller 包下所有类的所有方法,无论返回类型和参数类型是什么,都是我想切的对象

  • 连接点

    使用 JoinPoint 获取连接点信息,包括方法签名等

  • 通知

    使用 @Before@After 注解定义了两个通知方法,分别在方法执行前后执行

  • 织入

    在运行时,Spring AOP 将切面织入目标对象的方法中,从而在方法执行前后执行通知

总结

通过 Spring AOP,我们能够将关注点(如日志记录)与业务逻辑分离,使得代码更加清晰、可维护。在实际应用中,AOP 可以应用于诸如事务管理、安全性、性能监控等方面,为应用程序提供更高的可扩展性和可维护性。本文通过一个日志打印的例子,详细介绍了在 Spring AOP 中切面、切点、通知、连接点和织入的概念,并成功地使用 AOP 实现了在方法执行前后分别打印日志的功能。这种方式帮助我们实现了横切关注点的代码重用,提高了代码的可维护性和可读性,也希望通过本文的介绍,可以将 AOP 应用到具体项目开发中。

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

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

相关文章

【python入门】day17:模块化编程、math库常见函数

什么叫模块 模块的导入 导入所有&#xff1a;import 模块名称 导入指定&#xff1a;from 模块名称 import 函数/变量/类 python的math库 什么是math库 Python的math库是Python的内建库之一&#xff0c;它提供了许多数学函数&#xff0c;包括三角函数、对数函数、幂函数等&a…

迅为RK3588开发板使用 FFMpeg 进行推流

Debian/Ubuntu 系统使用以下命令安装 FFMpeg &#xff0c;如下图所示&#xff1a; apt-get install ffmpeg 使用 ifconfig 查看开发板 ip 为 192.168.1.245 如下图所示&#xff1a; 使用 FFMpeg 推流一个 mp4 视频进行测试&#xff0c;作者将测试视频 test.mp4 放在了根目录下…

linuxnodejs 20.* 安装问题,version `GLIBCXX_3.4.26‘

背景 今天服务器被重置拉&#xff0c;nodejs 环境不存在&#xff0c;特意安装下nodejs&#xff0c;一访问官网&#xff0c;妈呀&#xff0c;居然到20版本拉&#xff01;就尝试安装下最新版本&#xff01; 过程 $ cd /opt $ curl -OL https://nodejs.org/dist/v20.10.0/node-v2…

c++语言基础18-开房门

题目描述 假设你手里有一串钥匙&#xff0c;这串钥匙上每把钥匙都有一个编号&#xff0c;对应着一个房门的编号。现给你一个房门编号&#xff0c;你需要判断是否能够打开该房门。 输入描述 测试数据共有多组。 第一行为一个整数 s&#xff0c;表示共有多少组测试数据。 每组第一…

Spring高手之路-Spring中Bean的五大作用域

目录 Singleton&#xff08;单例&#xff09;&#xff1a;默认的作用域 Prototype&#xff08;原型&#xff09; Request&#xff08;请求&#xff09; Session&#xff08;会话&#xff09; Global Session&#xff08;全局会话&#xff09; 五大作用域范围对比 作用域…

[C#]基于deskew算法实现图像文本倾斜校正

【算法介绍】 让我们开始讨论Deskeweing算法的一般概念。我们的主要目标是将旋转的图像分成文本块&#xff0c;并确定它们的角度。为了让您详细了解我将使用的方法&#xff1a; 照常-将图像转换为灰度。应用轻微的模糊以减少图像中的噪点。现在&#xff0c;我们的目标是找到带…

【GoLang入门教程】Go语言几种标准库介绍(四)

编程语言的未来&#xff1f; 文章目录 编程语言的未来&#xff1f;前言几种库fmt库 (格式化操作)关键函数&#xff1a;示例 Go库标准库第三方库示例 html库(HTML 转义及模板系统)主要功能&#xff1a;示例 总结专栏集锦写在最后 前言 上一篇&#xff0c;我们介绍了debug、enco…

C/C++动态内存分配 malloc、new、vector(简单讲述)

路虽远&#xff0c;行则将至 事虽难&#xff0c;做则必成 今天来主要讲C中动态内存分配 其中会穿插一些C的内容以及两者的比较 如果对C语言中的动态内存分配还不够理解的同学 可以看看我之前的博客:C语言动态分配 在讲解C的动态内存分配之前 我们先讲一下C内存模型 &#xff1…

国家信息安全水平等级考试NISP二级题目卷③(包含答案)

国家信息安全水平等级考试NISP二级题目卷&#xff08;三&#xff09; 国家信息安全水平等级考试NISP二级题目卷&#xff08;三&#xff09;需要报考咨询可以私信博主&#xff01; 前言&#xff1a; 国家信息安全水平考试(NISP)二级&#xff0c;被称为校园版”CISP”,由中国信息…

全解析阿里云Alibaba Cloud Linux镜像操作系统

Alibaba Cloud Linux是基于龙蜥社区OpenAnolis龙蜥操作系统Anolis OS的阿里云发行版&#xff0c;针对阿里云服务器ECS做了大量深度优化&#xff0c;Alibaba Cloud Linux由阿里云官方免费提供长期支持和维护LTS&#xff0c;Alibaba Cloud Linux完全兼容CentOS/RHEL生态和操作方式…

78 Python开发-多线程FuzzWaf异或免杀爆破

这里写目录标题 本课知识点:学习目的:演示案例:Python开发-简单多线程技术实现脚本Python开发-利用FTP模块实现协议爆破脚本Python开发-配合Fuzz实现免杀异或Shell脚本 涉及资源: 本课知识点: 协议模块使用&#xff0c;Request爬虫技术&#xff0c;简易多线程技术&#xff0c;…

Postman 安装及使用

文章目录 1. 安装 Postman1&#xff09;下载2&#xff09;安装3&#xff09;注册用户4&#xff09;登陆完成 2. 创建和发送请求1&#xff09;发送一个 GET 请求2&#xff09;发送一个 POST 请求 3. 查看响应4. 使用环境变量和变量5. 高级功能和测试6. 导出和分享请求总结 Postm…

IDS 和 IPS:了解异同

IDS 和 IPS 是至关重要的网络安全技术&#xff0c;经常被混淆或互换使用。那么&#xff0c;IDS 和 IPS 之间有什么区别&#xff0c;哪一种是最适合您组织需求的选择呢&#xff1f; 什么是IDS&#xff08;入侵检测系统&#xff09;&#xff1f; 入侵检测系统 (IDS) 是一种网络…

git在本地创建dev分支并和远程的dev分支关联起来

文章目录 git在本地创建dev分支并和远程的dev分支关联起来1. 使用git命令2. 使用idea2.1 先删除上面建的本地分支dev2.2 通过idea建dev分支并和远程dev分支关联 3. 查看本地分支和远程分支的关系 git在本地创建dev分支并和远程的dev分支关联起来 1. 使用git命令 git checkout…

华芯微特|MCU之TIMER输入捕获

引言 华芯微特公司SWM系列单片机提供的TIMER个数和功能有些微差别&#xff0c;为了让您更加简单的使用这一功能&#xff0c;下面小编将以SWM190为例&#xff0c;我们今天详细讲解一下TIMER的输入捕获功能。 TIMER输入捕获 一、TIMER定时器之输入捕获功能 我们今天详细讲解一下…

一篇文章学会Linux

一篇文章学会Linux 声明&#xff1a;以下内容均为我个人的理解&#xff0c;如果发现错误或者疑问可以联系我共同探讨 简介 Linux Linux是一种自由和开放源码的类UNIX操作系统。该操作系统的内核由林纳斯托瓦兹在1991年10月5日首次发布&#xff0c;在加上用户空间的应用程序…

【JUC】Volatile关键字+CPU/JVM底层原理

Volatile关键字 volatile内存语义 1.当写一个volatile变量时&#xff0c;JMM会把该线程对应的本地内存中的共享变量值立即刷新回主内存中。 2.当读一个volatile变量时&#xff0c;JMM会把该线程对应的本地内存设置为无效&#xff0c;直接从主内存中读取共享变量 所以volatile…

运输层

title: 运输层 date: 2023-12-24 14:17:55 tags: 知识总结 categories: 计算机网络 运输层和网络层的联系和区别 物理层、数据链路层以及网络层它们共同解决了将主机通过异构网络互联起来所面临的问题&#xff0c;实现了主机到主机的通信&#xff0c;但实际上&#xff0c;在计…

视频监控可视化云平台EasyCVR智能视频技术优势分析

TSINGSEE青犀视频安防视频管理系统EasyCVR视频智能融合共享平台&#xff0c;是一个支持Windows/Linux(CentOS ubuntu)/国产化系统的视频管理平台。平台可以支持多协议接入&#xff0c;通过视频应用引擎将多种格式的视频数据转换为统一的视频流数据&#xff0c;支持无插件H5直播…

RocketMQ单机部署完整学习笔记

文章目录 前言一、RocketMQ是什么&#xff1f;二、使用步骤1.安装MQ1.安装JDK2.安装mq3.MQ配置(核心) 2.搭建可视化dashboard1.下载源码2.修改配置3.启动 3.整合java1.生产者2.消费者3.启动生产者4.启动消费者5.dashboard添加消费组 三、总结全部的配置 前言 本文是基于4.X版本…