简介

前面已经讲完 spring-bean( 详见Spring ),这篇博客开始攻克 Spring 的另一个重要模块--spring-aop。

spring-aop 可以实现动态代理(底层是使用 JDK 动态代理或 cglib 来生成代理类),在项目中,一般被用来实现日志、权限、事务等的统一管理。

我将通过两篇博客来详细介绍 spring-aop 的使用、源码等。这是第一篇博客,主要介绍 spring-aop 的组件、架构、使用等。

项目环境

maven:3.6.3

操作系统:win10

JDK:8u231

Spring:5.2.6.RELEASE

几个重要的组件

说到 spring-aop,我们经常会使用到**PointcutJoinpointAdviceAspect**等等基础组件,它们都是抽象出来的“标准”,有的来自 aopalliance,有的来自 AspectJ,也有的是 spring-aop 原创。

想要学好 spring-aop,必须理解好这几个基础组件。但是,理解它们非常难,一个原因是网上能讲清楚的不多,第二个原因是这些组件本身抽象得不够直观(spring 官网承认了这一点)。

AOP联盟的组件--Joinpoint、Advice

在 spring-aop 的包中内嵌了 aopalliance 的包(aopalliance 就是一个制定 AOP 标准的联盟、组织),这个包是 AOP 联盟提供的一套“标准”,提供了 AOP 一些通用的组件,包的结构大致如下。

└─org
└─aopalliance
├─aop
│ Advice.class
│ AspectException.class

└─intercept
ConstructorInterceptor.class
ConstructorInvocation.class
Interceptor.class
Invocation.class
Joinpoint.class
MethodInterceptor.class
MethodInvocation.class

完整的 aopalliance 包,除了 aop 和 intercept,还包括了 instrument 和 reflect,后面这两个部分 spring-aop 没有引入,这里就不说了。

使用 UML 表示以上类的关系,如下。可以看到,这主要包含两个部分:JoinpointAdvice

  1. Joinpoint一个事件,包括调用某个方法(构造方法或成员方法)、操作某个成员属性等

例如,我调用了user.study() 方法,这个事件本身就属于一个JoinpointJoinpoint是一个“动态”的概念,通过它可以获取这个事件的静态信息,例如当前事件对应的AccessibleObjectAccessibleObjectFieldMethodConstructor等的超类)。spring-aop 主要使用到Joinpoint的子接口MethodInvocation

  1. AdviceJoinpoint执行的某些操作

例如,JDK 动态代理使用的InvocationHandler、cglib 使用的MethodInterceptor,在抽象概念上可以算是Advice(即使它们没有继承Advice)。spring-aop 主要使用到Advice的子接口MethodInterceptor

  1. Joinpoint 和 Advice 的关系

JoinpointAdvice操作的对象,一个Advice可以操作多个Joinpoint,一个Joinpoint也可以被多个Advice操作。在 spring-aop 里,Joinpoint对象会持有一条Advice链,调用Joinpoint.proceed()将逐一执行其中的Advice(需要判断是否执行),执行完AdviceAdvice链,将最终执行被代理对象的方法

AspectJ 的组件--Pointcut、Aspect

AspectJ 是一个非常非常强大的 AOP 工具,可以实现编译期织入、编译后织入和类加载时织入,并且提供了一套 AspectJ 语法(spring-aop 支持这套语法,但要额外引入 aspectjweaver 包)。spring-aop 使用到了 AspectJ 的两个组件,PointcutAspect

其中,Pointcut可以看成一个过滤器,它可以用来判断当前Advice是否拦截指定Joinpoint,如下图所示。注意,不同的Advice也可以共用一个Pointcut

Aspect这个没什么特别,就是一组Pointcut+Advice的集合。下面这段代码中,有两个Advice,分别为printRequestprintResponse,它们共享同一个Pointcut,而这个类里的Pointcut+Advice可以算是一个Aspect

@Aspect
public class UserServiceAspect { private static final Logger LOGGER = LoggerFactory.getLogger(UserServiceAspect.class); @Pointcut("execution(* cn.zzs.spring.UserService+.*(..)))")
public void genericPointCut() { } @Before(value = "genericPointCut()")
public void printRequest(JoinPoint joinPoint) throws InterruptedException {
//······
} @After(value = "genericPointCut()")
public void printResponse(JoinPoint joinPoint) throws InterruptedException {
//······;
}
}

spring-aop 的组件--Advisor

Advisor是 spring-aop 原创的一个组件,一个Advice加上它对应的过滤器,就组成了一个Advisor。在上面例子中,printRequestAdvice加上它的Pointcut,就是一个Advisor。而Aspect由多个Advisor组成。

注意,这里的过滤器,可以是Pointcut,也可以是 spring-aop 自定义的ClassFilter

spring-aop 和 AspectJ 的关系

从 AOP 的功能完善程度来讲,AspectJ 支持编译期织入、编译后织入和类加载时织入,并且提供了一套 AspectJ 语法,非常强大。在 AspectJ 面前,spring-aop 就是个“小弟弟”。

spring-aop 之所以和 AspectJ 产生关联,主要是因为借鉴了 AspectJ 语法(这套语法一般使用注解实现,用于定义AspectPointcutAdvice等),包括使用到 AspectJ 的注解以及解析语法的类。如果我们希望在 spring-aop 中使用 AspectJ 注解语法,需要额外引入 aspectjweaver 包。

如何使用 spring-aop

接下来展示的代码可能有的人看了会觉得奇怪,“怎么和我平时用 spring-aop 不一样呢?”。这里先说明一点,因为本文讲的是 spring-aop,所以,我用的都是 spring-aop 的 API,而实际项目中,由于 spring 封装了一层又一层,导致我们感知不到 spring-aop 的存在。

通常情况下,Spring 是通过向BeanFactory注册BeanPostProcessor(例如,AbstractAdvisingBeanPostProcessor)的方式对 bean 进行动态代理,原理并不复杂,相关内容可以通过 spring-bean 了解( Spring源码系列(二)--bean组件的源码分析 )。

接下来让我们抛开这些“高级封装”,看看 spring-aop 的真面目。

spring-aop 的代理工厂

下面通过一个 UML 图来了解下 spring-aop 的结构,如下。

spring-aop 为我们提供了三种代理工厂,其中ProxyFactory比较普通,AspectJProxyFactory支持 AspectJ 语法的代理工厂,ProxyFactoryBean可以给 Spring IoC 管理的 bean 进行代理。

下面介绍如何使用这些代理工厂来获得代理类。

使用ProxyFactory生成代理类

ProxyFactory的测试代码如下,如果指定了接口,一般会使用 JDK 动态代理,否则使用 cglib。

    @Test
public void test01() { ProxyFactory proxyFactory = new ProxyFactory(); // 设置被代理的对象
proxyFactory.setTarget(new UserService());
// 设置代理接口--如果设置了接口,一般会使用JDK动态代理,否则用cglib
// proxyFactory.setInterfaces(IUserService.class); // 添加第一个Advice
proxyFactory.addAdvice(new MethodInterceptor() { public Object invoke(MethodInvocation invocation) throws Throwable {
TimeUnit.SECONDS.sleep(1); LOGGER.info("打印{}方法的日志", invocation.getMethod().getName());
// 执行下一个Advice
return invocation.proceed();
}
}); // 添加第二个Advice······ IUserService userController = (IUserService)proxyFactory.getProxy();
userController.save();
userController.delete();
userController.update();
userController.find();
}

运行以上方法,可以看到控制台输出:

2020-09-12 16:32:02.704 [main] INFO  cn.zzs.spring.ProxyFactoryTest - 打印save方法的日志
增加用户
2020-09-12 16:32:03.725 [main] INFO cn.zzs.spring.ProxyFactoryTest - 打印delete方法的日志
删除用户
2020-09-12 16:32:04.726 [main] INFO cn.zzs.spring.ProxyFactoryTest - 打印update方法的日志
修改用户
2020-09-12 16:32:05.726 [main] INFO cn.zzs.spring.ProxyFactoryTest - 打印find方法的日志
查找用户

使用ProxyFactoryBean生成代理类

ProxyFactoryBeanProxyFactory差不多,区别在于ProxyFactoryBean的 target 是一个 bean。因为需要和 bean 打交道,所以这里需要创建 beanFactory 以及注册 bean。另外,我们可以设置每次生成的代理类都不同。

    @Test
public void test01() {
// 注册bean
DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory();
beanFactory.registerBeanDefinition("userService",
BeanDefinitionBuilder.rootBeanDefinition(UserService.class).getBeanDefinition()); ProxyFactoryBean proxyFactory = new ProxyFactoryBean();
// 设置beanFactory
proxyFactory.setBeanFactory(beanFactory);
// 设置被代理的bean
proxyFactory.setTargetName("userService");
// 添加Advice
proxyFactory.addAdvice(new MethodInterceptor() { public Object invoke(MethodInvocation invocation) throws Throwable {
TimeUnit.SECONDS.sleep(1); LOGGER.info("打印{}方法的日志", invocation.getMethod().getName());
return invocation.proceed();
}
});
// 设置scope
//proxyFactory.setSingleton(true);
proxyFactory.setSingleton(false); IUserService userController = (IUserService)proxyFactory.getObject(); userController.save();
userController.delete();
userController.update();
userController.find(); IUserService userController2 = (IUserService)proxyFactory.getObject();
System.err.println(userController == userController2);
}

使用AspectJProxyFactory生成代理类

使用AspectJProxyFactory要额外引入 aspectjweaver 包,如下:

        <dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.6</version>
<!-- <scope>runtime</scope> -->
</dependency>

接下来配置一个Aspect,如下。这里定义了一个Advice,即 printRequest 方法;定义了一个Pointcut,即拦截UserService及其子类。

@Aspect
public class UserServiceAspect { private static final Logger LOGGER = LoggerFactory.getLogger(UserServiceAspect.class); @Pointcut("execution(* cn.zzs.spring.UserService+.*(..)))")
public void genericPointCut() { } @Before(value = "genericPointCut()")
public void printRequest(JoinPoint joinPoint) throws InterruptedException {
TimeUnit.SECONDS.sleep(1);
LOGGER.info("call {}_{} with args:{}",
joinPoint.getSignature().getDeclaringType().getSimpleName(),
joinPoint.getSignature().getName(),
joinPoint.getArgs());
}
}

编写生成代理类的方法,如下。AspectJProxyFactory会利用 AspectJ 的类来解析 Aspect,并转换为 spring-aop 需要的Advisor

    @Test
public void test01() { AspectJProxyFactory proxyFactory = new AspectJProxyFactory();
// 设置被代理对象
proxyFactory.setTarget(new UserService()); // 添加Aspect
proxyFactory.addAspect(UserServiceAspect.class); // 提前过滤不符合Poincut的类,这样在执行 Advice chain 的时候就不要再过滤
proxyFactory.setPreFiltered(true); UserService userController = (UserService)proxyFactory.getProxy(); userController.save();
userController.delete();
userController.update();
userController.find();
}

关于 spring-aop 的组件、架构、使用等内容,就介绍到这里,第二篇博客再分析具体源码。

感谢阅读。以上内容如有错误,欢迎指正。

相关源码请移步:spring-aop

本文为原创文章,转载请附上原文出处链接:https://www.cnblogs.com/ZhangZiSheng001/p/13671149.html

Spring源码系列(三)--spring-aop的基础组件、架构和使用的更多相关文章

  1. spring源码学习(三)--spring循环引用源码学习

    在spring中,是支持单实例bean的循环引用(循环依赖)的,循环依赖,简单而言,就是A类中注入了B类,B类中注入了A类,首先贴出我的代码示例 @Component public class Add ...

  2. Spring源码系列(一)--详解介绍bean组件

    简介 spring-bean 组件是 IoC 的核心,我们可以通过BeanFactory来获取所需的对象,对象的实例化.属性装配和初始化都可以交给 spring 来管理. 针对 spring-bean ...

  3. Spring源码系列(四)--spring-aop是如何设计的

    简介 spring-aop 用于生成动态代理类(底层是使用 JDK 动态代理或 cglib 来生成代理类),搭配 spring-bean 一起使用,可以使 AOP 更加解耦.方便.在实际项目中,spr ...

  4. AOP执行增强-Spring 源码系列(5)

    AOP增强实现-Spring 源码系列(5) 目录: Ioc容器beanDefinition-Spring 源码(1) Ioc容器依赖注入-Spring 源码(2) Ioc容器BeanPostProc ...

  5. Spring源码系列 — BeanDefinition扩展点

    前言 前文介绍了Spring Bean的生命周期,也算是XML IOC系列的完结.但是Spring的博大精深,还有很多盲点需要摸索.整合前面的系列文章,从Resource到BeanDefinition ...

  6. Spring源码系列 — Bean生命周期

    前言 上篇文章中介绍了Spring容器的扩展点,这个是在Bean的创建过程之前执行的逻辑.承接扩展点之后,就是Spring容器的另一个核心:Bean的生命周期过程.这个生命周期过程大致经历了一下的几个 ...

  7. Spring源码系列 — BeanDefinition

    一.前言 回顾 在Spring源码系列第二篇中介绍了Environment组件,后续又介绍Spring中Resource的抽象,但是对于上下文的启动过程详解并未继续.经过一个星期的准备,梳理了Spri ...

  8. 事件机制-Spring 源码系列(4)

    事件机制-Spring 源码系列(4) 目录: Ioc容器beanDefinition-Spring 源码(1) Ioc容器依赖注入-Spring 源码(2) Ioc容器BeanPostProcess ...

  9. Ioc容器依赖注入-Spring 源码系列(2)

    Ioc容器依赖注入-Spring 源码系列(2) 目录: Ioc容器beanDefinition-Spring 源码(1) Ioc容器依赖注入-Spring 源码(2) Ioc容器BeanPostPr ...

随机推荐

  1. Golang Gtk+3教程:Grid布局

    在上个例子中我们使用了box布局,现在让我们来学习另一种布局--grid.其实这几种布局都大同小异,如果你看懂了上一个例子,想必使用grid也不是难事. 程序运行效果: package main im ...

  2. java jsp实现网络考试系统(mysql)

    java网络考试系统 功能:可进行学生.管理员登录,学生考试.管理员出卷.列表分页 @ 目录 java网络考试系统 实现效果 主要代码实现 写在最后 实现效果 主要代码实现 package cn.it ...

  3. PDF的来源——概率密度函数

    //首发于简书,详见原文:https://www.jianshu.com/p/6493edd20d61 你不会还真的以为这是一篇讲怎么做pdf文件,怎么编辑.保存.美化的文章吧? 咳咳,很遗憾告诉你不 ...

  4. Android Studio 突然无法识别真机问题

    最近在赶项目,今天AS突然疯狂跟我作对,森气!! 平时连接手机没有问题,今天突然各种识别不到真机!! 1.数据线,check.没有问题. 2.重启AS,还是不行. 3.安装驱动,行不通. 4.已经弹出 ...

  5. GridBagConstraints详解

    名称 作用 默认值 常量 位置 gridx 行(x)的第一个单元格 0并且为非负数 RELATIVE(相对的) 紧跟前一个组件的后面 gridy 列(y)的第一个单元格 0并且为非负数 RELATIV ...

  6. char **指针动态分配地址空间

    1. 定义char **类型变量,初始化为NULL 2. 分配行指针数组空间 3. 为每行分配空间 4. 释放每行的空间 5. 释放二维指针 void main() { char **pointer= ...

  7. Docker系列——Docker安装&基础命令

    Docker 概述 Docker 是一个开源的应用容器引擎,Docker 可以让开发者打包他们的应用以及依赖包到一个轻量级.可移植的容器中,然后发布到任何流行的 Linux 机器上,也可以实现虚拟化. ...

  8. 团队作业1——团队展示&选题 (追忆少年)

    目录 一,团队展示 1.1队名 1.2队员学号 1.3项目描述 1.4队员风采 1.5团队分工 1.6团队合照 1.7团队特色 (一)目标导向 (二)协作基础 (三)共同的规范和方法 (四)技术或技能 ...

  9. 玩转Spring——Spring IOC/DI

    什么是IOCioc :Inversion of Control,即控制反转. 它不是一种技术,而是一种设计思想,即java程序中获取对象的方式发生反转,由最初的new方式创建,转变成由第三方框架创建. ...

  10. tensorflow1.x及tensorflow2.x不同版本实现验证码识别

    近一个假期,入坑深度学习,先从能看得着的验证码识别入门.从B站看了几天的黑马程序员的“3天带你玩转python深度学习后“,一是将教程中提到的代码一一码出来:二是针对不同的tensorflow版本,结 ...