Spring的AOP开发-注解方式开发AOP

news2025/1/23 4:54:46

基于注解配置的AOP

注解方式AOP的基本使用

  • Spring的AOP也提供了注解方式配置,使用相应的注解替代之前的xml配置,xml配置AOP时,我们主要配置了三部分:目标类被Spring容器管理(注解使用@Service)、通知类被Spring容器管理(注解使用@Component)、通知与切点的织入(切面)(使用注解@Aspect,不同的通知使用不同对应的注解),如下

使用注解具体示例代码

package com.example.advice;


import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

// 自定义增强类,内部提供增强方法
@Component
@Aspect // 告诉Spring容器该类是一个切面
public class MyAdvice {
    @Before("execution(void com.example.Service.ServiceImpl.UserServiceImpl.*(..))")
    // todo 前置通知
    public void beforeAdvice() {
        System.out.println("前置通知");

    }

    // todo 后置通知
    @AfterReturning("execution(void com.example.Service.ServiceImpl.UserServiceImpl.*(..))")
    public void afterAdvice() {
        System.out.println("后置通知");

    }

    // todo 环绕通知
    @Around("execution(void com.example.Service.ServiceImpl.UserServiceImpl.*(..))")
    public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        // 前置通知
        System.out.println("环绕通知:前置通知");
        Object res = proceedingJoinPoint.proceed(); // 执行目标方法
        System.out.println("环绕通知中目标方法执行了");
        // 后置通知
        System.out.println("环绕通知:后置通知");
        return res;
    }

    // todo 异常通知
    @AfterThrowing(pointcut = "execution(void com.example.Service.ServiceImpl.UserServiceImpl.*(..))", throwing = "throwable")
    public void afterThrowingAdvice(Throwable throwable) {
        System.out.println("异常抛出通知...出现异常才会执行");
    }

    // todo 最终通知
    @After("execution(void com.example.Service.ServiceImpl.UserServiceImpl.*(..))")
    public void endAdvice() {
        System.out.println("最终通知....怎么样都会通知");
    }

}

 只需要在配置文件中指定组件扫描范围即可

测试类代码如下

package com.example.Test;


import com.example.Service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class TestMyAOP {
    public static void main(String[] args) {
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext3.xml");
        UserService userService = applicationContext.getBean(UserService.class);
        userService.show1();
    }
}

 运行结果如下

可以参考java web专栏中的AOP相关文章:内容管理-CSDN创作中心

注解方式AOP配置详解

package com.example.advice;


import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.*;
import org.springframework.stereotype.Component;

// 自定义增强类,内部提供增强方法
@Component
@Aspect // 告诉Spring容器该类是一个切面
public class MyAdvice {
    // todo 切点表达式的抽取
    @Pointcut("execution(void com.example.Service.ServiceImpl.UserServiceImpl.*(..))")
    public void MyPointCut() {
    }

    @Before("MyAdvice.MyPointCut()")
    // todo 前置通知
    public void beforeAdvice() {
        System.out.println("前置通知");

    }

    // todo 后置通知
    @AfterReturning("MyAdvice.MyPointCut()")
    public void afterAdvice() {
        System.out.println("后置通知");

    }

    // todo 环绕通知
    @Around("MyAdvice.MyPointCut()")
    public Object around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
        // 前置通知
        System.out.println("环绕通知:前置通知");
        Object res = proceedingJoinPoint.proceed(); // 执行目标方法
        System.out.println("环绕通知中目标方法执行了");
        // 后置通知
        System.out.println("环绕通知:后置通知");
        return res;
    }

    // todo 异常通知
    @AfterThrowing(pointcut = "MyAdvice.MyPointCut()", throwing = "throwable")
    public void afterThrowingAdvice(Throwable throwable) {
        System.out.println("异常抛出通知...出现异常才会执行");
    }

    // todo 最终通知
    @After("MyAdvice.MyPointCut()")
    public void endAdvice() {
        System.out.println("最终通知....怎么样都会通知");
    }

}

同样的,可以使用配置类来代替上述的配置文件实现全注解开发

package com.example.Config;

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;

//注解配置类替代配置文件,实现纯注解开发
@Configuration // 表示该类是一个核心配置类,同时将该类交给Spring容器管理(内置了@Component注解)
@ComponentScan({"com.example"})//<context:component-scan base-package="com.example"/>
public class SpringConfig {

}

 测试类中的获取Spring容器的方法也要改变一下。

注解方式AOP原理解析

如果使用xml配置文件+注解的方式:

还是通过标签中的命名空间对应的命名空间处理器,在MATE-INF文件夹下的spring.handlers文件夹中,进行查找

 不同命名空间对应不同解析器,然后一步步追溯源码,最终还是落脚到实现BeanPostProcessor接口,重写其中的postProcessAfterInitialization方法,向容器中注册bean对象。

如果使用配置类的方式(全注解)

查看注解类中的关键注解@EnableAspectJAutoProxy,

导入了相关类,来注册bean对象


图示三种方法都是底层源码最中都是使用红色框中自动代理构造器创建proxy对象。 

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

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

相关文章

图像和视频上传平台Share Me

本文完成于 6 月&#xff0c;所以反代中&#xff0c;域名演示还是使用的 laosu.ml&#xff0c;不过版本并没有什么变化&#xff1b; 什么是 Share Me &#xff1f; Share Me 是使用 Next.js 和 PocketBase 的自托管图像和视频上传平台&#xff0c;具有丰富的嵌入支持和 API&…

基于Java的高校宿舍管理系统设计与实现(源码+lw+部署文档+讲解等)

文章目录 前言具体实现截图论文参考详细视频演示为什么选择我自己的网站自己的小程序&#xff08;小蔡coding&#xff09;有保障的售后福利 代码参考源码获取 前言 &#x1f497;博主介绍&#xff1a;✌全网粉丝10W,CSDN特邀作者、博客专家、CSDN新星计划导师、全栈领域优质创作…

【C++】unordered_map和unordered_set

哈希表 1. unordered_map1.1 概念1.2 常见接口 2. unordered_set2.1 概念2.1 常见接口 3. 底层实现3.1 哈希3.2 哈希函数3.3 闭散列和开散列3.3.1 闭散列3.3.2 开散列 3.4 模拟实现3.4.1 改造哈希桶3.4.2 模拟实现unordered_set3.4.3 模拟实现unordered_map 在C11中&#xff0c…

Promise, async, await 学习

异步编程简介&#xff1a; 介绍&#xff1a;异步编程是一种编程范式&#xff0c;旨在提高程序的性能和响应能力。该模型允许程序在执行某些任务时&#xff0c;不必等待这些任务完成就可以进行下一步操作&#xff0c;从而提高了程序的效率。 作用&#xff1a;异步编程通常用于…

IPT2602协议-USB 快速充电端口控制器

产品描述&#xff1a; IPT2602是一款USB端口快速充电协议控制芯片。IPT2602智能识别多种快速充电协议&#xff0c;对手机等受电设备进行快速充电。IPT2602根据受电设备发送的电压请求能够精确的调整VBUS输出电压&#xff0c;从而实现快速充电。 IPT2602在调整5V输出电压前会自动…

10.5 认识XEDParse汇编引擎

XEDParse 是一款开源的x86指令编码库&#xff0c;该库用于将MASM语法的汇编指令级转换为对等的机器码&#xff0c;并以XED格式输出&#xff0c;目前该库支持x86、x64平台下的汇编编码&#xff0c;XEDParse的特点是高效、准确、易于使用&#xff0c;它可以良好地处理各种类型的指…

[硬件基础]-快速了解RS232串行通信

快速了解RS232串行通信 文章目录 快速了解RS232串行通信1、概述2、什么是串行数据通信&#xff1f;3、什么是RS232&#xff1f;4、RS232应用5、RS232如何工作&#xff1f;6、RS232协议基础6.1 电压与逻辑表示6.2 数据编码6.3 起始位和停止位6.4 奇偶校验位6.5 波特率6.5 RS232电…

掌握 SwiftUI 中的 ScrollView

文章目录 前言scrollTransition 修饰符ScrollTransitionPhase弹性动画总结 前言 SwiftUI 框架的第五个版本引入了许多与 ScrollView 相关的新 API&#xff0c;使其比以前更强大。本周将开始介绍 ScrollView 在 SwiftUI 中的新功能系列文章&#xff0c;首先我们将讨论滚动过渡。…

【LeetCode: 918. 环形子数组的最大和 | 动态规划】

&#x1f680; 算法题 &#x1f680; &#x1f332; 算法刷题专栏 | 面试必备算法 | 面试高频算法 &#x1f340; &#x1f332; 越难的东西,越要努力坚持&#xff0c;因为它具有很高的价值&#xff0c;算法就是这样✨ &#x1f332; 作者简介&#xff1a;硕风和炜&#xff0c;…

关联规则挖掘(上):数据分析 | 数据挖掘 | 十大算法之一

⭐️⭐️⭐️⭐️⭐️欢迎来到我的博客⭐️⭐️⭐️⭐️⭐️ &#x1f434;作者&#xff1a;秋无之地 &#x1f434;简介&#xff1a;CSDN爬虫、后端、大数据领域创作者。目前从事python爬虫、后端和大数据等相关工作&#xff0c;主要擅长领域有&#xff1a;爬虫、后端、大数据…

【全方位带你配置yolo开发环境】快速上手yolov5

本文用于记录yolo开发环境的配置&#xff0c;以及我在配置中出现的各种问题&#xff0c;以供大伙参考。&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; 本人持续分享更多关于电子通信专业内容以及嵌入式和单片机的知识…

阿里云服务器ECS详细介绍_云主机_服务器托管_弹性计算

阿里云服务器ECS英文全程Elastic Compute Service&#xff0c;云服务器ECS是一种安全可靠、弹性可伸缩的云计算服务&#xff0c;阿里云提供多种云服务器ECS实例规格&#xff0c;如经济型e实例、通用算力型u1、ECS计算型c7、通用型g7、GPU实例等&#xff0c;阿里云服务器网分享阿…

使用css制作3D盒子,目的是把盒子并列制作成3D货架

1. 首先看效果&#xff08;第一个五颜六色的是透明多个面&#xff0c;第2-3都是只有3个面是我实际需要的&#xff0c;右边的有3个并列的正方体与3个并列的长方体&#xff09;&#xff1a; 长方体与正方体&#xff0c;所有代码&#xff1a; <!DOCTYPE html> <html lan…

汽车网络安全--安全芯片应用场景解析

​在聊汽车网络安全时,最先想到的就是使用芯片内置HSM,比如说英飞凌TC2xx系列的HSM、瑞萨RH850的ICU、NXP的HSE等等;实际上除了内置HSM,还有外置HSM(通过UART、SPI等通信)、安全存储芯片等等。而这些芯片统称为安全芯片。 安全芯片的主要作用是为整个系统建立起一个可信的…

【Java】类和接口的区别

1. 类和类的继承关系&#xff08;一个类只能单继承一个父类&#xff0c;不能继承n多个不同的父类&#xff09; 继承关系&#xff0c;只能单继承&#xff0c;但可以多层继承 2. 类和接口的实现关系&#xff08;一个类可以实现n多个不同的接口&#xff09; 实现关系&#xff0c;可…

【QT开发笔记-基础篇】| 第四章 事件QEvent | 4.4 鼠标按下、移动、释放事件

本章要实现的整体效果如下&#xff1a; QEvent::MouseButtonPress ​ 鼠标按下时&#xff0c;触发该事件&#xff0c;它对应的子类是 QMouseEvent QEvent::MouseMove ​ 鼠标移动时&#xff0c;触发该事件&#xff0c;它对应的子类是 QMouseEvent QEvent::MouseButtonRel…

golang gin框架1——简单案例以及api版本控制

gin框架 gin是golang的一个后台WEB框架 简单案例 package mainimport ("github.com/gin-gonic/gin""net/http" )func main() {r : gin.Default()r.GET("/ping", func(c *gin.Context) {//以json形式输出&#xff0c;还可以xml protobufc.JSON…

网络安全黑客究竟是什么?

“网络安全”是指任何活动旨在保护您的网络和数据的可用性和完整性。它包括硬件和软件技术。有效的网络安全管理对网络的访问。它针对的是一种不同的威胁,阻止他们进入或在您的网络传播。 网络安全是如何工作的呢? 网络安全结合多层防御的优势和网络。每个网络安全层实现政策…

前端TypeScript学习day01-TS介绍与TS常用类型

(创作不易&#xff0c;感谢有你&#xff0c;你的支持&#xff0c;就是我前行的最大动力&#xff0c;如果看完对你有帮助&#xff0c;请留下您的足迹&#xff09; 目录 TypeScript 介绍 TypeScript 是什么 TypeScript 为什么要为 JS 添加类型支持&#xff1f; TypeScript 相…

【Redis】基础数据结构-quicklist

Redis List 在Redis3.2版之前&#xff0c;Redis使用压缩列表和双向链表作为List的底层实现。当元素个数比较少并且元素长度比较小时&#xff0c;Redis使用压缩列表实现&#xff0c;否则Redis使用双向链表实现。 ziplist存在问题 不能保存过多的元素&#xff0c;否则查找复杂度…