【SSM-笔记】Spring AOP 详讲,面向切面编程

news2024/11/23 7:47:05

Spring的AOP实现

  • 👀AOP的概述
    • 引入 AOP
  • 👀面向切面编程之七大术语
  • 👀切点表达式(重要)
    • 切入点表达式的语法格式
  • 👀AOP的实现
    • AspectJ概述
    • 五大通知测试
    • 切面顺序(@Order注解)
    • 通用切点(@Pointcut注解)
    • 全注解形式
  • 👀XML配置实现
  • 👀总结

绪论:

AOP最好了解代理模式(静态代理,动态代理(JDK动态代理、CGLIB动态代理))后再去学会如何使用,我觉得可能效率会比较高吧。
还有就是本篇大部分都是老杜的Spring笔记,少部分自身理解。
代理模式可以看看篇:代理模式,挺全的。
结尾我也会给出老杜的 Spring 笔记。

👀AOP的概述

AOP(Aspect-Oriented-Programing)面向切面编程。它是Spring的重要思想之一。

引入 AOP

小编就知道,Java是一种面向对象(OOP,Object-Oriented-Programing)的编程语言。使用静态代理去处理拓展业务功能时就能很好的体现出来。当我们使用继承去实现拓展某个业务的时候(比如有时候业务需要记录日志,需要统计什么东西,需要事务管理时),代码会存在冗余,而且因为存在继承关系,所以存在高耦合,而且业务操作不是单个的,有多少个业务操作,就要有多少个重复代码(记录日志,统计,事务管理等)。显然是不利于维护的。

(可能有人说将这些公告代码写成方法不就好了?但是不是还是需要程序员去调用吗,所以小编觉得仍不够好,仍然存在重复代码。)

于是有了AOP,AOP将权限校验、日志记录等非业务代码完全提取出来,与业务代码分离,并寻找结点切入到代码中。

在这里插入图片描述

👀面向切面编程之七大术语

  • 连接点 Joinpoint
    在程序的整个执行流程中,可以织入切面的位置,方法的执行前后,异常抛出之后等位置连接点描述的是位置,切入点。

  • 切点 Pointcut
    在程序执行流程中,真正织入切面的方法(一个切点可以对应多个连接点)。切点描述的是方法。(在哪个方法上切,描述的就是这个)

  • 通知 Advice
    通知又叫做增强,就是具体要织入的代码通知描述的是代码。
    通知包括:

    1. 前置通知(方法前)@Before
    2. 后置通知(方法后)@AfterReturning
    3. 环绕通知(方法前后都有)@Around
    4. 异常通知(放在 catch 语句块中)@AfterThrowing
    5. 最终通知(放在 finally 语句块中)@After
  • 切面 Aspect
    切点 + 通知就是切面。(方法加上通知要织入的代码形成的面及为切面)

  • 织入 Weaving
    把通知应用到目标对象的过程。

  • 代理对象 Proxy
    一个目标对象被织入通知后产生的新对象。

  • 目标对象 Target
    被织入通知的对象。

(代理对象和目标对象是指代理模式中的相应角色)

下面很好的描述AOP四个主要的术语:

在这里插入图片描述
我觉得这样理解七大术语算具体的了,看官方或者书籍资料给的解释还得花时间去理解概念。

在这里插入图片描述

👀切点表达式(重要)

切点肯定是需要指定的,可以通过 表达式或者模式 指定切入点 Pointcut。

切点表达式即用来定义通知(Advice)往哪些方法上切入。

切入点表达式的语法格式

execution([访问控制权限修饰符] 返回值类型 [全限定类名]. 方法名(形式参数列表) [异常])
[] 中的内容属于可选项

  • 访问控制权限修饰符(可选项,没写就是4个权限都包括)

  • 返回值类型(必填项,* 表示返回值类型任意)

  • 全限定类名(可选项,“…”代表当前包以及子包下的所有类,省略则表示所有的类)

  • 方法名(必填项,* 表示所有的方法,set*表示所有的set方法)

  • 形式参数列表

  1. 必填项;
  2. ()表示没有参数列表;
  3. (. .)表示参数类型和个数随意的方法;
  4. (*)表示只有一个参数的方法;
  5. ( *,String)表示第一个参数随意,第二个参数必须是String类型的。
  • 异常(可选项,省略时表示任意异常类型)

注意:返回值类型、方法名、形式参数列表是必填项。类名和方法是点连接起来的。

举例:

这个表示:service包下的所有公开的delete方法(参数和返回值任意)
execution(public * com.powernode.mall.service.*.delete*(..))
这个表示:mall 包下的所有类的所有方法
execution(* com.powernode.mall.*(..))
这个表示:所有类的所有方法
execution(* *(..))

👀AOP的实现

Spring常用的AOP实现有俩种方式:

  1. Spring框架结合AspectJ框架实现的AOP,基于注解的方式。(使用较普遍)
  2. Spring框架结合AspectJ框架实现的AOP,基于XML方式。

AspectJ概述

Eclipse组织的一个支持AOP的框架。AspectJ框架是独立于Spring框架之外的一个框架,Spring框架用了AspectJ。

五大通知测试

首先你想要利用Spring去实现AOP,那么目标对象,和代理所需执行的通知对象肯定是要暴露给Spring的,不然它咋实现呢?所以这一过程中仍然是需要使用DI的。

为实现AOP,最重要的就是确定好切点和通知切面=切点+通知。可以使用AspectJ框架下的注解@Aspect对切面类进行标注,以提示Spring该类存在很多个切面实现。

切点可以利用切点表达式进行确认,通知即是要写的代码,而切入点的位置就是通知的位置,用那通知五大注解即可(@Before、@AfterReturning、@Around、@AfterThrowing、@After

测试先提供个目标类

@Service
public class OrderService {
	// 测试用来做切面
    public void order(){
        System.out.println("正在进行订单");
    }
}

测试程序

	@Test
    public void testAopByAnnotation(){
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("spring.xml");
        OrderService orderService = applicationContext.getBean("orderService", OrderService.class);
        orderService.order();
    }
  • @Before测试
@Component
@Aspect // 切面类是需要使用@Aspect注解进行标注的。
public class LogAspect {// 切面

    private final Logger logger = Logger.getLogger(Logger.class);

    @Before("execution(* com.ncpowernode.spring6.service.OrderService.order(..))")
    public void beforeAdvice(){
        logger.info("订单之前进行的操作~");
    }

}

在这里插入图片描述

  • @AfterReturning
@AfterReturning("execution(* com.ncpowernode.spring6.service.OrderService.order(..))")
    public void afterAdvice(){
        logger.info("订单之后进行的操作");
    }

在这里插入图片描述

  • @Around

环绕通知需要传入一个连接点对象 ProceedingJoinPoint,以便告诉Spring 哪个是切点前的通知,哪个是后的。

@Around("execution(* com.ncpowernode.spring6.service.OrderService.order(..))")
    public void around(ProceedingJoinPoint joinPoint) throws Throwable {
        // 前通知
        logger.info("订单前");
        // 切点程序
        joinPoint.proceed();
        // 后通知
        logger.info("订单后");
    }

在这里插入图片描述

  • @AfterThrowing

该通知对应的是catch语句块;

下面给OrderService类中添加段语句测试一下

在这里插入图片描述

@AfterThrowing("execution(* com.ncpowernode.spring6.service.OrderService.order(..))")
    public void exceptionAdvice(){
        logger.error("出现异常");
    }

在这里插入图片描述

  • @After
@After("execution(* com.ncpowernode.spring6.service.OrderService.order(..))")
    public void finallyAdvice(){
        logger.info("finally内执行语句");
    }

在这里插入图片描述

可以简单推断代理过程是 try-catch-finally 的形式的。

切面顺序(@Order注解)

日常项目中可能不止存在一个切面类,比如可能会有日志切面类、安全切面类、事务控制切面类等等。但如果业务中同时使用俩个及以上的切面类呢?执行顺序是怎样的呢?

下面是@Order注解的源代码,里面有个int类型的value属性,默认值是Integer.MAX_VALUE整型最大值。

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.TYPE, ElementType.METHOD, ElementType.FIELD})
@Documented
public @interface Order {
    int value() default Integer.MAX_VALUE;
}

通过@Target元注解可以知道,该@Order注解可以出现的位置。

LogAspect 切面类 + SecurityAspect 切面类进行测试。

下面是SecurityAspect 类的代码

@Aspect
@Component
@Order(1)
public class SecurityAspect {

    private final Logger logger = Logger.getLogger(SecurityAspect.class);

    @Before("execution(* com.ncpowernode.spring6.service.OrderService.order(..))")
    public void beforeAdvice(){
        logger.info("安全前置通知");
    }

}

下面是LogAspect 类的代码

@Component
@Aspect // 切面类是需要使用@Aspect注解进行标注的。
@Order(2)
public class LogAspect {

	private final Logger logger = Logger.getLogger(Logger.class);

	@Before("execution(* com.ncpowernode.spring6.service.OrderService.order(..))")
    public void beforeAdvice(){
        logger.info("订单之前进行的操作");
    }
}

@Order 注解中 value 值小的先执行。

通用切点(@Pointcut注解)

为了使得通用表达式得到复用,可以使用@Pointcut注解让某个方法表示通用切点表达式。

下面是@Pointcut注解的源码。(通过@Target元注解可以知道仅能用在方法上)

@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface Pointcut {
    String value() default "";

    String argNames() default "";
}

测试

在这里插入图片描述

在另一个切面类中也可以使用通用切点表达式。

在这里插入图片描述

在这里插入图片描述

全注解形式

这里需要使用三个注解(都只能用在类上):

  1. @Configuration注解(用来替代xml文件)
  2. @ComponentScan注解(用来扫描注解,其中value属性值是指定的扫描包)
  3. @EnableAspectJAutoProxy(使用AspectJ框架实现AOP,可以设置proxyTargetClass 属性值为true,表示只用CGLIB动态代理)

类名无所谓,下面举例

@Configuration
@ComponentScan("com.ncpowernode.spring6.service")
@EnableAspectJAutoProxy(proxyTargetClass = true)
public class Spring6Config {
}

测试程序,现在实例化的是AnnotationConfigApplicationContext对象了,然后向上转型为ApplicationContext对象。

@Test
    public void testNoXml(){
        ApplicationContext applicationContext = new AnnotationConfigApplicationContext(Spring6Config.class);
        OrderService orderService = applicationContext.getBean("orderService", OrderService.class);
        orderService.order();
    }

测试结果

在这里插入图片描述

👀XML配置实现

不管是注解配置还是xml配置文件的方式进行配置,都是根据下面的步骤:

  1. 首先是确定目标对象,然后确定切面类(同时排好切面类的执行顺序,用@Order注解)
  2. 将目标类和切面类(用@Aspect注解标注了的)暴露给Spring容器。
  3. 确定切面,切面=切点+通知。切点利用切点表达式进行设置,通知即对应着切面类中的方法。

下面给出个案例配置然后执行测试

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
                            http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd
                            http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">

    <bean id="registeService" class="com.ncpowernode.spring6.xml.service.RegisteService"/>
    <bean id="registeAspect" class="com.ncpowernode.spring6.xml.service.RegisteAspect"/>

    <aop:config>
        <!--切点表达式-->
        <aop:pointcut id="mypointcut" expression="execution(* com.ncpowernode.spring6.xml.service.RegisteService.registe(..))"/>
        <!--切面:通知 + 切点-->
        <aop:aspect ref="registeAspect">
            <aop:around method="around" pointcut-ref="mypointcut"/>
        </aop:aspect>
    </aop:config>


</beans>

在这里插入图片描述

👀总结

老杜的Spring笔记
密码:mg9b

还借用了骆驼整理说博客中的图片。骆驼整理说是小编最近比较喜欢的一位博主。
骆驼整理说的主页

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

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

相关文章

Python uiautomator2安卓自动化测试

一、前言 uiautomator2是Python对Android设备进行UI自动化的库&#xff0c;支持USB和WIFI链接&#xff0c;可以实现获取屏幕上任意一个APP的任意一个控件属性&#xff0c;并对其进行任意操作。 重点是它可以实现安卓自动化采集&#xff0c;甚至是群控采集&#xff0c;且安装和…

【CSS】快速入门笔记

视频链接&#xff1a;https://www.bilibili.com/video/BV1mS4y1Z7Ga/?spm_id_from333.999.0.0&vd_source1ad00d913eae8281cbadad6ae66fb06c 文章目录一、CSS语法1.结构2.样式类型1&#xff09;内联样式 Inline Style2&#xff09;内部样式 Internal Style3&#xff09;外部…

【深度强化学习】(4) Actor-Critic 模型解析,附Pytorch完整代码

大家好&#xff0c;今天和各位分享一下深度强化学习中的 Actor-Critic 演员评论家算法&#xff0c;Actor-Critic 算法是一种综合了策略迭代和价值迭代的集成算法。我将使用该模型结合 OpenAI 中的 Gym 环境完成一个小游戏&#xff0c;完整代码可以从我的 GitHub 中获得&#xf…

Docker 安装mysql Mac 环境下

已安装桌面端 Docker &#xff08;Mac安装Docker&#xff09; 安装方式一 打开链接 https://www.docker.com/products/docker-desktop 选择平台下载 安装方式二 安装homebrew /usr/bin/ruby -e "$(curl -fsSL https://raw.githubusercontent.com/Homebrew/install/m…

Parcel Bundle漏洞学习

Bundle的序列化细节看上去还是有些复杂的&#xff0c;在之前已经讨论过&#xff0c;一般我们使用Parcel的时候&#xff0c;都是严格的write和read相对应。一些疏漏&#xff0c;不对应&#xff0c;竟然就可以成为漏洞&#xff0c;https://xz.aliyun.com/t/2364 里介绍了Bundle漏…

多态的定义、重写、原理

多态 文章目录多态多态的定义和条件协变&#xff08;父类和子类的返回值类型不同&#xff09;函数隐藏和虚函数重写的比较析构函数的重写关键字final和override抽象类多态的原理单继承和多继承的虚函数表单继承下的虚函数表多继承下的虚函数表多态的定义和条件 定义&#xff1…

add_header重写的坑

问题描述&#xff1a; nginx 的 add_header 配置在很多文档中都标注为&#xff1a;“可以覆盖响应头”&#xff0c;然而并没有说出使用场景&#xff0c;导致不少开发人员在使用 add_header 时都出现了错误&#xff1a;add_header 根本没有重写响应头&#xff01; add_header 的…

指向函数的指针详解,以及如何使用指向函数的指针变量做函数参数

指向函数的指针作为函数参数&#xff0c;是 C 语言实际应用中的一个比较深入的部分。 目录 一、什么是函数的指针 二、用函数指着变量调用函数 2.1举例说明 三、怎样定义和使用指向函数的指针变量 3.1定义指向函数的指针变量 3.2指向函数的指针变量详解 3.3通过指针变量…

httpd使用记录

httpd使用记录 Busybox用一个httpd的程序&#xff0c;尝试用起来。 简单测试 启动服务 # 启动服务 mkdir /var/www/html httpd -p 8080 -h /var/www/html &编写html文件 在/var/www/html下放一个测试网页index.html文件。 <!DOCTYPE html> <html><hea…

2022济南大学acm新生赛题解

通过答题情况的难度系数&#xff1a; 签到&#xff1a;A 简单&#xff1a;BL 中等&#xff1a;D 困难&#xff1a;CM 极难&#xff1a;KNO A-和 算出n个数的和判断正负性即可&#xff01;&#xff01;&#xff01; 发现很多同学的代码错误&#xff1a;要么sum未赋初值&…

数据库--进阶篇--9--存储引擎

MySQL体系结构 索引是在引擎层&#xff0c;所以不同的存储引擎&#xff0c;它的索引结构不同。 存储引擎简介 存储引擎就是存储数据、建立所以、更新/查询数据等技术的实现方式。存储引擎是基于表的&#xff0c;而不是基于库的&#xff0c;所以存储引擎也可以被称为表类型。 …

如何利用多功能智慧杆建设森林防火监测预警网?

近年来&#xff0c;我国森林面积稳步增长&#xff0c;生态空间质量稳中向好&#xff0c;森林碳汇能力稳步提升&#xff0c;“绿水青山就是金山银山”的发展理念得到有效贯彻。但随着森林面积的不断扩大&#xff0c;森林火灾的风险也随之增长&#xff0c;森林防火的工作也尤为重…

JS学习第9天——ES6中面向对象(类class、constructor构造函数、类的继承extends、super关键字、面向对象tab栏切换案例)

目录一、面向对象1、面向过程2、面向对象3、两者对比二、ES6中的类和对象1、面向对象的思维特点2、对象3、类class4、类constructor构造函数三、类的继承1、继承2、super()关键字3、注意点四、面向对象案例一、面向对象 两大编程思想&#xff1a;① 面向过程 ② 面向对象 1、…

Python源码解读(一):CPython源码下载和参考资料

1、写在前面 大家好&#xff0c;这里是程序员晚枫&#xff0c;全平台同名。 之前在广东工作&#xff0c;从事后端开发&#xff0c;主要使用Java和Python&#xff0c;自己业余时间做了一些Python的开源项目。 来重庆工作以后&#xff0c;经过曲折的求职阶段&#xff0c;目前从…

网络、接口学习笔记一

Gigabit Ethernet 是标识 千兆以太网网口&#xff0c;10GE就是万兆&#xff0c;100GE就是10万兆。 VLAN是什么&#xff1f; VXLAN是什么&#xff1f; 局域网是什么&#xff1f; RJ45接口是什么&#xff1f; GE电口和RJ45透传口的区别&#xff1f; VLAN&#xff08;虚拟局域网…

超超超超保姆式详解——字符函数和字符串函数(学不会打我)上

目录 长度不受限制的字符串函数 strlen部分 strlen函数的易错小知识 strlen函数的实现 strcpy部分 strcat部分 自己实现strcat strstr函数部分 简单例子&#xff1a; 分析 strcmp部分 长度受限制的字符串函数 strncpy 简单例子 strncat strncmp 简单例子 &…

深度学习知识点简单概述【更新中】

文章目录人工神经网络的定义神经元的定义神经元的功能单层神经网络感知机人工神经网络的定义 人工神经网络(英语:Artificial Neural Network&#xff0c;ANN)&#xff0c;简称神经网络(Neural Network,NN&#xff09;或类神经网络&#xff0c;是一种模仿生物神经网络(动物的中…

class03:MVVM模型与响应式原理

目录一、MVVM模型二、内在1. 深入响应式原理2. Object.entries3. 底层搭建一、MVVM模型 MVVM&#xff0c;即Model 、View、ViewModel。 Model > data数据 view > 视图&#xff08;vue模板&#xff09; ViewModel > vm > vue 返回的实例 > 控制中心, 负责监听…

docker基本命令 - 数据卷

作用 ● 做数据持久化。防止容器一旦停止运行&#xff0c;该容器中运行产生的数据就没了 ● 不同容器之间的数据共享(大鲸鱼背上各个小集装箱之间可以共享数据) 交互式命令使用 docker run -it -v / 宿主机的绝对路径目录:/容器内绝对路径目录 镜像名 docker run -it -v / 宿…

SUBMIT的用法

SUBMIT的用法 一、简介 系统MB52/MB51/MB5B等类似的报表 &#xff0c;虽然数据很全面&#xff0c;执行效率也够快&#xff0c;但是经常会不满足用户需求&#xff08;增添字段、添加查询条件等&#xff09;&#xff0c;很多ABAP 会选择去COPY出标准程序&#xff0c;然后去做修改…