【SSH快速进阶】——Spring AOP原理及其实现

简介:   上篇文章简单总结了一下静态代理以及JDK和Cglib两种方式实现的动态代理。由此也延伸出了一个重要的编程思想:AOP。

 上篇文章简单总结了一下静态代理以及JDK和Cglib两种方式实现的动态代理。由此也延伸出了一个重要的编程思想:AOP。


AOP原理


 AOP(Aspect Oriented Programming):面向切面编程。利用AOP可以对业务逻辑中的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。


 AOP主要面向独立的服务,跟业务没有任何关系,通常会遍布在系统的任何角落,如系统的日志、权限检查等服务。


 在Spring 的AOP中,有如下概念:


39.png



 上图中,每部分的解释如下:


 Cross cutting concern

 横切性关注点,一种独立服务,它会遍布在系统的处理流程中(比如日志服务、权限检查等)


 Aspect

 对横切性关注点的模块化(如果横切服务要运行,就必须放在一个类里,可以把它看做是Aspect)


 Advice

 对横切性关注点的具体实现(包括before、after、throwing等),比如在用户操作之前检查用户权限(before);在生成订单之后记录日志(after);在某个方法抛异常之后进行异常处理(throwing)等。


 Pointcut

 它定义了Advice应用到哪些JoinPoint上(即应用到哪些类的那些方法上),对Spring来说是方法调用。动态代理中,会把服务加到所有方法上,但在实际应用中可能不会把服务加到所有的方法上,AOP中就可以定义切入点,即进行过滤,AOP服务加到哪些方法上。


 JoinPoint

 Advice在应用程序上执行的点或时机(比如在类A的a()方法执行之前执行某个操作,或修改类A的b属性时触发某个操作),Spring只支持方法的JoinPoint,Aspecj可以支持可以使属性修改的JoinPoint。Advice执行在JoinPoint上。



Spring AOP实现


 Spring AOP实现步骤大致如下:


 1、Spring jar包的依赖


SPRING_HOME/dist/spring.jar
SPRING_HOME/lib/log4j/log4j-1.2.14.jar
SPRING_HOME/lib/jakarta-commons/commons-logging.jar
SPRING_HOME/lib/aspectj/*.jar


  2、首先在Spring配置文件中启用Aspect对Annotation的支持:


<aop:aspectj-autoproxy/>


  3、定义Aspect


import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.AfterThrowing;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Before;
@Aspect
public class SecurityHandler {
  @Pointcut("execution(* add*(..))")
  private void addMethod(){}
  /**
   * 在【* add*(..)】匹配的方法执行之前执行
   */
  @Before("addMethod()")
  private void checkSecurityBeforeAdd(){
    System.out.println("----Before Add:checkSecurity-----");
  }
  /**
   * 在【* add*(..)】匹配的方法执行之后执行
   */
  @After("addMethod()")
  private void addLogAfterAdd(){
    System.out.println("----After Add:addLog-----");
  }
  /**
   * 在【* add*(..)】匹配的方法执行报异常之后执行
   */
  @AfterThrowing("addMethod()")
  private void addLogAfterThrowing(){
    System.out.println("----After Throwing:Throwing-----");
  }
}


 上面的SecurityHandler 类就是权限检查的服务,可以定义在某些方法执行之前、某些方法执行之后等时间点切入该服务。


 @Aspect

 在SecurityHandler类上打上@Aspect注解,就说明它是上面图中的Cross cutting concern经过封装后的模块了,Spring容器加载该类的时候会把它里面定义的所有横切性服务(Advice)都应用到相应的方法(Pointcut)上。


 @Pointcut

 上面例子中@Pointcut标记的方法名称是addMethod,且该方法是一个没有参数和返回值的空方法,该方法可以看做就是一种标记,不进行调用。("execution(* add*(..))") 定义的就是过滤要添加横切性服务的一些方法的条件。具体解释是:第一个*匹配所有类型返回值的方法;第二个*匹配所有方法头带“add”的方法;..表示匹配有参数和无参数的方法都满足条件。更多过滤条件请见Spring Framework开发手册,小编也特意为大家截了张图:点这里看大图


 @Before、@After、@Throwing

 上面例子中其余三种Advice:@Before、@After、@Throwing,大家看注释应该也就看明白了,这是三个执行时机,分别会在目标方法执行前、执行后、报异常后执行。需要注意的是注解后面括号里的内容必须要是@Pointcut定义的方法名称,说明这三个时机执行的Advice都为该方法服务。



 3、客户端调用AOP 服务


public static void main(String[] args) {
  String[] configurations=new String[]{"applicationContext-bean.xml"};
  beanFactory=new ClassPathXmlApplicationContext(configurations);
  UserManager userManager=(UserManager)beanFactory.getBean("userManager");
  userManager.addUser(new User("Danny","12456")); 
}


  以上是用注解的形式来配置的AOP服务,当然也可以用配置文件来配,用配置文件配的时候,上面第2步骤可以省略。

  代理类:


public class SecurityHandler{
  private void checkSecurity(){
    System.out.println("----checkSecurity-----");
  }
}


  以上是用注解的形式来配置的AOP服务,当然也可以用配置文件来配,用配置文件配的时候,上面第2步骤可以省略。

  代理类:

public class SecurityHandler{
  private void checkSecurity(){
    System.out.println("----checkSecurity-----");
  }
}


  配置文件:


<bean id="securityHandler" class="com.danny.spring.SecurityHandler"/>
<aop:config>
  <aop:aspect id="securityAspect" ref="securityHandler">
    <aop:pointcut id="addMethod" expression="execution(* add*(..))"/>
    <aop:before method="checkSecurity" pointcut-ref="addMethod"/>
  </aop:aspect>
</aop:config>


  如果把这种思想运用到项目中,与业务不相干的服务都用AOP切入,极大地减小了服务与业务的耦合,提升了开发效率。

相关文章
|
22天前
|
Java 关系型数据库 数据库
深度剖析【Spring】事务:万字详解,彻底掌握传播机制与事务原理
在Java开发中,Spring框架通过事务管理机制,帮我们轻松实现了这种“承诺”。它不仅封装了底层复杂的事务控制逻辑(比如手动开启、提交、回滚事务),还提供了灵活的配置方式,让开发者能专注于业务逻辑,而不用纠结于事务细节。
|
5月前
|
存储 人工智能 自然语言处理
RAG 调优指南:Spring AI Alibaba 模块化 RAG 原理与使用
通过遵循以上最佳实践,可以构建一个高效、可靠的 RAG 系统,为用户提供准确和专业的回答。这些实践涵盖了从文档处理到系统配置的各个方面,能够帮助开发者构建更好的 RAG 应用。
2531 114
|
2月前
|
缓存 安全 Java
Spring 框架核心原理与实践解析
本文详解 Spring 框架核心知识,包括 IOC(容器管理对象)与 DI(容器注入依赖),以及通过注解(如 @Service、@Autowired)声明 Bean 和注入依赖的方式。阐述了 Bean 的线程安全(默认单例可能有安全问题,需业务避免共享状态或设为 prototype)、作用域(@Scope 注解,常用 singleton、prototype 等)及完整生命周期(实例化、依赖注入、初始化、销毁等步骤)。 解析了循环依赖的解决机制(三级缓存)、AOP 的概念(公共逻辑抽为切面)、底层动态代理(JDK 与 Cglib 的区别)及项目应用(如日志记录)。介绍了事务的实现(基于 AOP
|
2月前
|
监控 架构师 NoSQL
spring 状态机 的使用 + 原理 + 源码学习 (图解+秒懂+史上最全)
spring 状态机 的使用 + 原理 + 源码学习 (图解+秒懂+史上最全)
|
4月前
|
前端开发 Java 数据库连接
Spring核心原理剖析与解说
每个部分都是将一种巨大并且复杂的技术理念传达为更易于使用的接口,而这就是Spring的价值所在,它能让你专注于开发你的应用,而不必从头开始设计每一部分。
164 32
|
4月前
|
Java 开发者 Spring
Spring框架 - 深度揭秘Spring框架的基础架构与工作原理
所以,当你进入这个Spring的世界,看似一片混乱,但细看之下,你会发现这里有个牢固的结构支撑,一切皆有可能。不论你要建设的是一座宏大的城堡,还是个小巧的花园,只要你的工具箱里有Spring,你就能轻松搞定。
184 9
|
10月前
|
XML Java 开发者
Spring Boot开箱即用可插拔实现过程演练与原理剖析
【11月更文挑战第20天】Spring Boot是一个基于Spring框架的项目,其设计目的是简化Spring应用的初始搭建以及开发过程。Spring Boot通过提供约定优于配置的理念,减少了大量的XML配置和手动设置,使得开发者能够更专注于业务逻辑的实现。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,为开发者提供一个全面的理解。
174 0
|
5月前
|
安全 前端开发 Java
Spring Boot 项目中触发 Circular View Path 错误的原理与解决方案
在Spring Boot开发中,**Circular View Path**错误常因视图解析与Controller路径重名引发。当视图名称(如`login`)与请求路径相同,Spring MVC无法区分,导致无限循环调用。解决方法包括:1) 明确指定视图路径,避免重名;2) 将视图文件移至子目录;3) 确保Spring Security配置与Controller路径一致。通过合理设定视图和路径,可有效避免该问题,确保系统稳定运行。
336 0
|
9月前
|
NoSQL Java Redis
Spring Boot 自动配置机制:从原理到自定义
Spring Boot 的自动配置机制通过 `spring.factories` 文件和 `@EnableAutoConfiguration` 注解,根据类路径中的依赖和条件注解自动配置所需的 Bean,大大简化了开发过程。本文深入探讨了自动配置的原理、条件化配置、自定义自动配置以及实际应用案例,帮助开发者更好地理解和利用这一强大特性。
1367 15
|
10月前
|
Java 开发者 Spring
Spring AOP 底层原理技术分享
Spring AOP(面向切面编程)是Spring框架中一个强大的功能,它允许开发者在不修改业务逻辑代码的情况下,增加额外的功能,如日志记录、事务管理等。本文将深入探讨Spring AOP的底层原理,包括其核心概念、实现方式以及如何与Spring框架协同工作。