AOP相关概念

在学习AOP实现原理之前,先了解下AOP相关基础知识。

AOP面向切面编程,它可以通过预编译方式或者基于动态代理对我们编写的代码进行拦截(也叫增强处理),在方法执行前后可以做一些操作,一般我们会看到以下几个概念:

连接点(JointPoint): AOP进行切入的位置称为连接点,一般指程序中的某个方法,对该方法进行拦截

通知(Advice): 在某个连接点执行的操作称为通知,也就是被拦截方法执行前后需要执行的操作称为通知,一共有五种

  • 前置通知:作用于被拦截方法执行之前
  • 后置通知:作用于被拦截方法执行之后进行的操作,无论被拦截方法是否抛出异常都会执行
  • 环绕通知:作用于被拦截方法执行之前和执行之后
  • 返回通知:作用于被拦截方法正常执行完毕返回时,如果抛出异常将不会执行
  • 异常通知:作用于被拦截方法抛出异常时

切点(Pointcut): 切点作用在于让程序知道需要在哪个连接点(方法)上执行通知,所以它也可以是一个表达式,匹配所有需要拦截的方法。

切面(Aspect): 切点通知共同组成了切面,其中切点定义了需要在哪些连接点上执行通知,通知里面定义了具体需要进行的操作

织入(Weaving):将切面连接到应用程序类型或者对象上,创建一个被通知的对象(advised object)的过程称为织入,换句话说织入就是将切面应用到目标对象的过程,它可以在编译期时(使用AspectJ)、加载时或者在运行时实现,Spring AOP是在运行时基于动态代理实现的。

Advisor:它是对切面的封装,使用了@AspectJ注解的类会被封装成Advisor。

Spring AOP和AspectJ区别

Spring AOP

Spring AOP是基于动态代理实现拦截功能的,默认使用JDK动态代理实现,当然这需要目标对象实现接口,如果目标对象没有实现接口,则使用CGLIB生成。

AspectJ

AspectJ提供了三种方式实现AOP:

  • 编译时织入:在编译期间将代码进行织入到目标类的class文件中。

  • 编译后织入:在编译后将代码织入到目标类的class文件中。

  • 加载时织入:在JVM加载class文件的时候进行织入。

Spring AOP的应用

了解了AOP相关知识后我们来实现一个需求:

  1. 自定义一个日志注解MyLogger
  2. 对使用了MyLogger注解的方法进行拦截,在方法的执行前后分别进行一些操作(环绕通知):
    • 方法执行前打印方法传入的参数
    • 方法执行后打印方法的返回值

自定义注解

import java.lang.annotation.*;

@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyLogger { }

定义切面Aspect

这里使用注解@Aspect来标记这是一个切面,切面是切点和通知的集合,分别使用注解@Pointcut和@Around实现。

@Slf4j
@Aspect // 使用注解定义切面
@Component
@EnableAspectJAutoProxy // 开启AOP
public class MyLogAspect { }
切点Pointcut

使用表达式@annotation(com.demo.mybatis.annotation.MyLogger)匹配所有使用了@MyLogger注解的方法。

    /**
* 定义切点,匹配所有使用了@MyLogger注解的方法
*/
@Pointcut("@annotation(com.example.annotation.MyLogger)") // 这里传入MyLogger的全路径
public void logPoiontcut() { }
通知Advice

定义一个logAroudAdvice方法,使用@Around注解标记这是一个环绕通知,logPoiontcut()引用了切点,表示通知要作用于哪些连接点上,该方法需要传入一个ProceedingJoinPoint类型参数(连接点):

    /**
* 通知Advice,这里使用了环绕通知
* @param joinPoint 连接点
* @return
* @throws Throwable
*/
@Around("logPoiontcut()") // 引用切点
public Object logAroudAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
// 方法执行前的日志打印
printBeforeLog(joinPoint);
// 执行方法
Object returnValue = joinPoint.proceed();
// 方法执行后的日志打印
printAfterLog(returnValue);
return returnValue;
}

完整的切面如下:

import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.stereotype.Component; import java.lang.reflect.Method; @Slf4j
@Aspect // 使用注解定义切面
@Component
@EnableAspectJAutoProxy
public class MyLogAspect { /**
* 定义切点,匹配所有使用了@MyLogger注解的方法
*/
@Pointcut("@annotation(com.example.annotation.MyLogger)")
public void logPoiontcut() {
} /**
* 通知Advice,这里使用了环绕通知
* @param joinPoint 连接点
* @return
* @throws Throwable
*/
@Around("logPoiontcut()") // 引用切点
public Object logAroudAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
// 方法执行前的日志打印
printBeforeLog(joinPoint);
// 执行方法
Object returnValue = joinPoint.proceed();
// 方法执行后的日志打印
printAfterLog(returnValue);
return returnValue;
} /**
* 方法执行前的日志打印
* @param joinPoint
*/
public void printBeforeLog(ProceedingJoinPoint joinPoint) {
MethodSignature methodSignature = (MethodSignature) joinPoint.getSignature();
// 获取方法
Method method = methodSignature.getMethod();
log.info("开始执行方法:{}", method.getName());
// 获取参数
Object[] args = joinPoint.getArgs();
if (args == null || args.length == 0) {
return;
}
// 获取参数名称
String[] parameterNames = methodSignature.getParameterNames();
StringBuilder parameterBuilder = new StringBuilder();
for (int i = 0; i < args.length; i++) {
parameterBuilder.append(parameterNames[i]).append(":").append(args[i]);
if (i < parameterNames.length - 1) {
parameterBuilder.append(",");
}
}
log.info("方法参数【{}】", parameterBuilder.toString());
} /**
* 方法执行后的日志打印
* @param returnValue
*/
public void printAfterLog(Object returnValue) {
log.info("方法返回值【{}】", returnValue == null ? null : returnValue.toString());
}
}

测试

定义一个用于计算的Service,实现一个两数相加的方法addTwoNum,并使用@MyLogger注解,对方法进行拦截,在方法执行前后打印相关日志

@Slf4j
@Service
public class ComputeService { // 使用自定义日志注解对方法进行拦截
@MyLogger
public Integer addTwoNum(Integer value1, Integer value2) {
log.info("执行addTwoNum方法");
return value1 + value2;
}
}

编写单元测试:

	@Autowired
private ComputeService computeService; @Test
public void testAddTwoNum {
computeService.addTwoNum(1, 2);
}

由于ComputeService没有实现接口,可以看到Spring默认使用了CGLIB生成对象:

日志输出如下,可以看到方法执行前后打印了相关日志:

开始执行方法:addTwoNum
方法参数【value1:1,value2:2】
执行addTwoNum方法
方法返回值【3】

参考

Spring官方文档

【 FatalFlower】AspectJ 简介

【Spring】AOP实现原理(一):AOP基础知识的更多相关文章

  1. Spring学习记录5——数据库事务基础知识

    何为数据库事务 “一荣共荣,一损共损”这句话很能体现事务的思想,很多复杂的事务要分步进行,但它们组成了一个整体,要么整体生效,要么整体失效.这种思想反映到数据库上,就是多条SQL语句,要么全部成功,要 ...

  2. 手写spring事务框架, 揭秘AOP实现原理。

    AOP面向切面编程:主要是通过切面类来提高代码的复用,降低业务代码的耦合性,从而提高开发效率.主要的功能是:日志记录,性能统计,安全控制,事务处理,异常处理等等. AOP实现原理:aop是通过cgli ...

  3. spring5 源码深度解析----- Spring事务 是怎么通过AOP实现的?(100%理解Spring事务)

    此篇文章需要有SpringAOP基础,知道AOP底层原理可以更好的理解Spring的事务处理. 自定义标签 对于Spring中事务功能的代码分析,我们首先从配置文件开始人手,在配置文件中有这样一个配置 ...

  4. JAVA基础加强(张孝祥)_类加载器、分析代理类的作用与原理及AOP概念、分析JVM动态生成的类、实现类似Spring的可配置的AOP框架

    1.类加载器 ·简要介绍什么是类加载器,和类加载器的作用 ·Java虚拟机中可以安装多个类加载器,系统默认三个主要类加载器,每个类负责加载特定位置的类:BootStrap,ExtClassLoader ...

  5. Spring 基础知识(一)基本概念 DI、IOC、AOP

    DI(依赖注入) 和IOC(控制反转)都是一种设计思想,要理解他们,让我们从coding中的一些痛点入手. 依赖注入 Dependency Injection : 如果A类要使用B类的一个方法,首先必 ...

  6. springmvc 运行原理 Spring ioc的实现原理 Mybatis工作流程 spring AOP实现原理

    SpringMVC的工作原理图: SpringMVC流程 . 用户发送请求至前端控制器DispatcherServlet. . DispatcherServlet收到请求调用HandlerMappin ...

  7. Spring框架系列(11) - Spring AOP实现原理详解之Cglib代理实现

    我们在前文中已经介绍了SpringAOP的切面实现和创建动态代理的过程,那么动态代理是如何工作的呢?本文主要介绍Cglib动态代理的案例和SpringAOP实现的原理.@pdai Spring框架系列 ...

  8. 【Spring】Spring AOP实现原理

    Spring AOP实现原理 在之前的一文中介绍过Spring AOP的功能使用,但是没有深究AOP的实现原理,今天正好看到几篇好文,于是就自己整理了一下AOP实现的几种方式,同时把代理模式相关知识也 ...

  9. Spring核心框架 - AOP的原理及源码解析

    一.AOP的体系结构 如下图所示:(引自AOP联盟) 层次3语言和开发环境:基础是指待增加对象或者目标对象:切面通常包括对于基础的增加应用:配置是指AOP体系中提供的配置环境或者编织配置,通过该配置A ...

随机推荐

  1. Linux利用crontab创建计划任务详解

    crontab 周期性计划任务 cron是Linux下的定时执行工具,可以在无需人工干预的情况下运行作业. 当需要周期性地重复执行任务时可以使用cron服务:该服务每分钟检查一次,并执行符合条件的任务 ...

  2. docker容器编排原来这么丝滑~

    前言: 请各大网友尊重本人原创知识分享,谨记本人博客:南国以南i 概念介绍: Docker Docker 这个东西所扮演的角色,容易理解,它是一个容器引擎,也就是说实际上我们的容器最终是由Docker ...

  3. 自定义user表签发token、自定义认证类、simpleui模块使用

    今日内容概要 自定义User表,签发token 自定义认证类 simpleui的使用 多方式登陆接口(后面也写 内容详细 1.自定义User表,签发token # 如果项目中的User表使用auth的 ...

  4. 简易table form梳理

    <!--      A:表格-table    <双标签,day3上午第一次接触>         作用:显示信息     一:table简易案例:         <tabl ...

  5. 迷惑小错 之 :requests.exceptions.ProxyError

    缘由 当打开代理或者抓包工具时 pycharm运行发包请求报错: requests.exceptions.ProxyError.关掉代理后又能正常的请求,这样对于我们日常操作很不方便吗.四处查找资料无 ...

  6. 1903021116-吉琛-Java第四周作业-程序编写

    项目 内容 课程班级博客链接 19级信计班 这个作业要求链接 Java分支语句学习 https://edu.cnblogs.com/campus/pexy/19xj/homework/12563 我的 ...

  7. vue2.x版本中Object.defineProperty对象属性监听和关联

    前言 在vue2.x版本官方文档中 深入响应式原理 https://cn.vuejs.org/v2/guide/reactivity.html一文的解释当中,Object.defineProperty ...

  8. [AcWing 823] 排列

    点击查看代码 #include<iostream> using namespace std; const int N = 10; int n; void dfs(int u, int nu ...

  9. 羽夏 MakeFile 简明教程

    写在前面   此系列是本人一个字一个字码出来的,包括示例和实验截图.该文章根据 GNU Make Manual 进行汉化处理并作出自己的整理,一是我对 Make 的学习记录,二是对大家学习 MakeF ...

  10. 伪元素选择器,选择器优先级,CSS修改文字属性,CSS修改字体属性,CSS修改其他属性

    伪元素选择器 未使用元素选择器的效果 第一行:伪元素选择器:选择部分内容 第二行:伪元素选择器:选择部分内容 伪元素选择器:选择部分内容 伪元素选择器:选择部分内容 ::selection:选择指定元 ...