热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

SpringAOP的简介与使用

什么是面向切面编程AOP是一种编程范式,并非是Spring的独创,它与语言无关,是一种编程思想。AOP可以帮助我们将通用的逻辑从业务逻辑中

什么是面向切面编程

AOP 是一种编程范式,并非是 Spring 的独创,它与语言无关,是一种编程思想。AOP 可以帮助我们将通用的逻辑从业务逻辑中分离出来。从项目角度来讲,有一些逻辑散布于程序的各个地方,但它们的逻辑又是千篇一律的相同。比如:我们需要打印每个请求的 URL、请求参数、IP地址,最简单的做法是在每一个 Controller 中去打印出这些信息。如果使用 AOP,我们能够更加轻松的处理这些问题。打印日志并不是使用 AOP 的唯一场景,我们还可以通过切面去统一处理 声明式事务、缓存、应用安全 等等。

AOP 相关术语

AOP 作为一种编程范式,已经衍生出了术语它的一些相关术语,例如:切面、通知、连接点、织入 等等。为了更好的理解如何在 Spring 中使用 AOP,我们必须对这些术语有一定的认知。

通知(Advice)

Spring AOP 支持五种类型的通知,它们分别定义了切面在什么时候使用,以及定义了切面需要做些什么。

  • @Before 前置通知,目标方法被调用之前执行
  • @After 后置通知,目标方法完成之后执行
  • @AfterReturning 返回通知,目标方法执行成功(未抛出异常)之后执行
  • @AfterThrowing 异常通知,目标方法执行失败(抛出异常)之后执行
  • @Around 环绕通知,目标方法执行前后都会调用

连接点(JoinPoint)

程序执行的某个特定位置:如类开始初始化前、类初始化后、类某个方法调用前、调用后、方法抛出异常后。一个类或一段程序代码拥有一些具有边界性质的特定点,这些点中的特定点就称为 "连接点"。Spring 仅支持方法的连接点,即仅能在方法调用前、方法调用后、方法抛出异常时以及方法调用前后这些程序执行点织入增强。连接点由两个信息确定:第一是用方法表示的程序执行点,第二是用相对点表示的方位。

切点(Poincut)

每个程序类都拥有多个连接点,如一个拥有两个方法的类,这两个方法都是连接点,即连接点是程序类中客观存在的事物。AOP 通过 "切点" 定位特定的连接点。连接点相当于数据库中的记录,而切点相当于查询条件。切点和连接点不是一对一的关系,一个切点可以匹配多个连接点。在 Spring 中,切点通过 org.springframework.aop.Pointcut 接口进行描述,它使用类和方法作为连接点的查询条件,Spring AOP 的规则解析引擎负责切点所设定的查询条件,找到对应的连接点。其实确切地说,不能称之为查询连接点,因为连接点是方法执行前、执行后等包括方位信息的具体程序执行点,而切点只定位到某个方法上,所以如果希望定位到具体连接点上,还需要提供方位信息。

切面(Aspect)

切面由切点和增强(引介)组成,它既包括了横切逻辑的定义,也包括了连接点的定义,Spring AOP 就是负责实施切面的框架,它将切面所定义的横切逻辑织入到切面所指定的连接点中。

引入(Introduction)

引入使我们具备了为类添加一些属性和方法的能力。这样,即使一个业务类原本没有实现某个接口,通过 AOP 的引介功能,我们可以动态地为该业务类添加接口的实现逻辑,让业务类成为这个接口的实现类。

织入(Weaving)

织入是将增强添加对目标类具体连接点上的过程。AOP 像一台织布机,将目标类、增强或引介通过 AOP 这台织布机天衣无缝地编织到一起。根据不同的实现技术,AOP 有三种织入的方式:1. 编译期织入,这要求使用特殊的Java编译器。2. 类装载期织入,这要求使用特殊的类装载器。3. 动态代理织入,在运行期为目标类添加增强生成子类的方式。Spring 采用动态代理织入,而 AspectJ 采用编译期织入和类装载期织入。

代理(Proxy)

一个类被 AOP 织入增强后,就产出了一个结果类,它是融合了原类和增强逻辑的代理类。根据不同的代理方式,代理类既可能是和原类具有相同接口的类,也可能就是原类的子类,所以我们可以采用调用原类相同的方式调用代理类。

AspectJ 切点表达式

Spring 只支持方法级别的连接点,切点表达式可以通过 execution 来定义。比如:@Pointcut(value = "execution(* com.vincent.springframework.example.Controller.*(..))") 表示匹配 com.vincent.springframework.example.Controller 这个类下面全部的方法。 关于 AspectJ 的 execution 表达式的详细信息,可以查看这篇文章 Spring AspectJ 的 execution表达式

添加 AsepctJ 的依赖

<dependency><groupId>org.aspectjgroupId><artifactId>aspectjweaverartifactId><version>1.9.1version>
dependency>

使用 XML 定义切面

在配置文件中创建切面&#xff0c;需要通过 Spring 的 aop 命名空间来开启。


<aop:aspectj-autoproxy/>

<content:component-scan base-package&#61;"com.vincent.spring.framework.example"/>

<bean id&#61;"messageService" class&#61;"com.vincent.spring.framework.example.MessageService"/>

<bean id&#61;"messageAspect" class&#61;"com.vincent.spring.framework.example.MessageAspect"/>

<aop:config><aop:aspect id&#61;"messageAspect" ref&#61;"messageAspect"><aop:before method&#61;"beforeAction" pointcut&#61;"execution(* com.vincent.spring.framework.example.MessageService.call(..))"/><aop:before method&#61;"afterAction" pointcut&#61;"execution(* com.vincent.spring.framework.example.MessageService.call(..))"/>aop:aspect>
aop:config>

使用 Java 配置定义切面

&#64;Configuration
&#64;EnableAspectJAutoProxy
&#64;ComponentScan(basePackage &#61; "com.vincent.spring.framework.example")
public class AspectJConfig {
}

&#64;Aspect
public class MessageAspect {// 目标方法执行之前&#64;Pointcut("execution(* com.vincent.spring.framework.example.MessageService.call(..))")public void beforeAction() {System.out.println("before action");}// 目标方法执行之后&#64;Pointcut("execution(* com.vincent.spring.framework.example.MessageService.call(..))")public void afterAction() {System.out.println("after action");}
}



推荐阅读
  • 在尝试启动Java应用服务器Tomcat时,遇到了org.apache.catalina.LifecycleException异常。本文详细记录了异常的具体表现形式,并提供了有效的解决方案。 ... [详细]
  • Hibernate全自动全映射ORM框架,旨在消除sql,是一个持久层的ORM框架1)、基础概念DAO(DataAccessorOb ... [详细]
  • PHP面试题精选及答案解析
    本文精选了新浪PHP笔试题及最新的PHP面试题,并提供了详细的答案解析,帮助求职者更好地准备PHP相关的面试。 ... [详细]
  • Flutter 核心技术与混合开发模式深入解析
    本文深入探讨了 Flutter 的核心技术,特别是其混合开发模式,包括统一管理模式和三端分离模式,以及混合栈原理。通过对比不同模式的优缺点,帮助开发者选择最适合项目的混合开发策略。 ... [详细]
  • 在Java开发中,保护代码安全是一个重要的课题。由于Java字节码容易被反编译,因此使用代码混淆工具如ProGuard变得尤为重要。本文将详细介绍如何使用ProGuard进行代码混淆,以及其基本原理和常见问题。 ... [详细]
  • Linux环境下MySQL的安装与配置
    本文详细介绍了在Linux系统上安装和配置MySQL的步骤,包括安装前的准备工作、下载和解压安装包、初始化数据库、配置文件编辑、启动服务以及设置开机自启动等。 ... [详细]
  • 为何Compose与Swarm之后仍有Kubernetes的诞生?
    探讨在已有Compose和Swarm的情况下,Kubernetes是如何以其独特的设计理念和技术优势脱颖而出,成为容器编排领域的领航者。 ... [详细]
  • Maven + Spring + MyBatis + MySQL 环境搭建与实例解析
    本文详细介绍如何使用MySQL数据库进行环境搭建,包括创建数据库表并插入示例数据。随后,逐步指导如何配置Maven项目,整合Spring框架与MyBatis,实现高效的数据访问。 ... [详细]
  • 本文介绍了SELinux的两种主要工作模式——强制模式和宽容模式,并提供了如何在CentOS 7中正确启用和配置SELinux的方法,以及在遇到登录问题时的解决策略。 ... [详细]
  • CentOS下ProFTPD的安装与配置指南
    本文详细介绍在CentOS操作系统上安装和配置ProFTPD服务的方法,包括基本配置、安全设置及高级功能的启用。 ... [详细]
  • 本文记录了在Windows 8.1系统环境下,使用IIS 8.5和Visual Studio 2013部署Orchard 1.7.2过程中遇到的问题及解决方案,包括503服务不可用错误和web.config配置错误。 ... [详细]
  • protobuf 使用心得:解析与编码陷阱
    本文记录了一次在广告系统中使用protobuf进行数据交换时遇到的问题及其解决过程。通过这次经历,我们将探讨protobuf的特性和编码机制,帮助开发者避免类似的陷阱。 ... [详细]
  • 本文探讨了在Windows系统中运行Apache服务器时频繁出现崩溃的问题,并提供了多种可能的解决方案和建议。错误日志显示多个子进程因达到最大请求限制而退出。 ... [详细]
  • 本文详细介绍了如何在Apache Kafka中进行Topic级别的配置,包括创建、修改和删除配置参数的具体步骤,并提供了详细的配置属性表。 ... [详细]
  • PCIe中的弹性缓冲原理解析
    PCIe是一种高速串行总线,其传输信号为差分信号,并采用同步传输方式。然而,PCIe并没有专用的同步时钟。本文将详细介绍PCIe中的弹性缓冲(Elastic Buffer)原理,探讨其如何处理时钟差异,确保数据传输的稳定性和可靠性。 ... [详细]
author-avatar
小老特
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有