动力节点-Spring使用AspectJ实现AOP前置通知

简介: AspectJ 是一个面向切面的框架,它扩展了 Java 语言。AspectJ 定义了 AOP 语法,它有一个专门的编译器用来生成遵守 Java 字节编码规范的 Class 文件。

AspectJ 是一个面向切面的框架,它扩展了 Java 语言。AspectJ 定义了 AOP 语法,它有一个专门的编译器用来生成遵守 Java 字节编码规范的 Class 文件。

AspetJ 是 Eclipse 的开源项目,官网介绍如下:

  • a seamless aspect-oriented extension to the Javatm programming language(一种基于 Java 平台的面向切面编程的语言)
  • Java platform compatible(兼容 Java 平台,可以无缝扩展)
  • easy to learn and use(易学易用)

AspectJ 中常用的通知类型有五种:

前置通知、后置通知、环绕通知、异常通知、最终通知

接下来这篇文章讲一下Spring框架使用AspectJ时实现AOP前置通知的相关知识:

相关视频教程:https://www.bilibili.com/video/BV1nz4y1d7uy

一、什么是AOP?


AOP面向切面编程,是基于动态代理的。AOP就是动态代理的规范化,把动态代理的实习按步骤和方式都定义好了,让开发人员用一种统一的方式使用动态代理。

Aspect Orient Programming:面向切面编程

Aspect:切面。给目标类增加的功能就是切面。添加的日志信息和事务都是切面。

切面的最大特点就是:非业务方法,和我们的业务功能没有关系,可以独立使用的。

二、怎么理解面向切面编程:


  1. 需要在分析项目时,找出切面。
  2. 合理的安排切面的执行时间(是在目标方法前执行,还是在目标方法后执行)。
  3. 合理的安排切面执行的位置(在哪个类,哪个方法增加增强功能)。

三、术语:


  1. Aspect:切面,表示要给业务方法增强的功能,就是一堆代码,这堆代码完成某一个功能。切面是非业务功能(日志,事务,统计信息,参数检查,权限验证)
  2. JoinPoint:连接点,连接业务方法和切面的位置,就是某个类中的业务方法。
  3. PointCut:切入点,指多个连接点方法的结合。是多个方法,这些方法都要加入切面的功能。
  4. 目标对象:要给哪个类增加方法
  5. Advice:通知,表示切面功能的执行时间(切面在什么时候执行,是在方法之前还是在方法之后)

四、一个切面有3个关键的要素:


  1. 切面的功能代码:切面要干什么
  2. 切面的执行位置,使用pointcut表示
  3. 切面的执行时间:在目标方法之前还是在目标方法之后,用advice表示

AOP的实现:是一个规范,是对动态代理的一个规范化。

五、AOP的技术实现框架:


  1. Spring:Spring实现了AOP规范,主要在事务处理时使用了AOP。项目开发中很少使用Spring的AOP实现。
  2. aspectJ:一个开源的专门做AOP的框架。Spring框架中集成了aspectJ框架,通过Spring就能使用aspectJ框架的功能了。aspectJ框架实现AOP有两种方式:
  • 使用xml配置文件
  • 使用注解(我们在项目中要做AOP功能,一般都使用注解),aspectJ有5个注解。

六、学习aspectJ框架的使用:


1、切面的执行时间,也就是advice通知。在aspectJ框架中是使用注解来表示的。

  • @Before:
  • @AfterReturning
  • @Around
  • @AfterThrowing
  • @After

2、表示切面执行的位置,使用的是切入点表达式。

七、execution (访问权限 方法返回值 方法声明(参数)异常类型)


其中,访问权限和异常类型这两个可以省略。

切入点表达式要匹配的对象就是目标方法的方法名字。

举例:

execution(public **(..)):指定切入点为任意公共方法

execution(set(..)):指定切入点为任何一个以set开始的方法

execution(com.xyz.service..*(..)):指定切入点为com.xyz.service 包中的任意类中的任意方法

八、使用aspectJ框架实现aop


使用aop的目的是给已经存在的类和方法在不改变原来的代码的情况下增加额外的功能。

九、使用aspectJ实现AOP的基本步骤:


  1. 新建maven项目
  2. 加入依赖:(1)Spring依赖(2)aspectJ依赖(3)Junit 单元测试
  3. 创建目标类:这个类有接口和接口的实现类。我们想要给这个类中的方法增加功能。
  4. 创建切面类(增加功能的代码写在切面类里面):普通类
  • 在类的上面加入@Aspect注解
  • 在类中定义切面方法,方法就是切面要执行的功能代码。在方法上面加入aspectJ中的通知注解,例如@Before,还需要指定切入点表达式。
    5.要把对象交给容器去创建,由容器Spring管理这些对象。
    创建Spring的配置文件,在配置文件中声明对象,把对象交给容器统一管理。声明对象可以使用注解或者xml的标签。
  • 声明目标对象
  • 声明切面类对象
  • 声明aspectJ框架中的自动代理生成器标签。 自动代理生成器:用来完成代理对象的自动创建功能的。
    6.创建测试类,从Spring容器中获取目标对象(这个对象实际就是代理对象)。通过代理执行方法,实现AOP的功能增强。

十、完整代码:


IDEA-Maven-模板选择quickstart-pom.xml添加相关依赖:

(1)Spring依赖

(2)aspectJ依赖

(3)Junit 单元测试

图片.png

<properties>
  <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  <maven.compiler.source>1.8</maven.compiler.source>
  <maven.compiler.target>1.8</maven.compiler.target>
</properties>
<dependencies>
  <!--单元测试-->
  <dependency>
    <groupId>junit</groupId>
    <artifactId>junit</artifactId>
    <version>4.12</version>
    <scope>test</scope>
  </dependency>
  <!--Spring依赖-->
  <dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-context</artifactId>
    <version>5.2.5.RELEASE</version>
  </dependency>
  <!--aspectJ依赖-->
  <dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-aspects</artifactId>
    <version>5.2.5.RELEASE</version>
  </dependency>
</dependencies>

右侧Maven查看添加的依赖

图片.png

Com.hik包下面的类和接口:

SomeService接口

package com.hik;
public interface SomeSeivece {
    void doSome(String name, Integer age);
}

SomeService接口的实现类SomeServiceImpl

package com.hik;
public class SomeSeiveceImpl implements SomeSeivece {
    @Override
    public void doSome(String name, Integer age) {
        //给doSome()方法增加一个功能,在doSome执行之前,输出方法的执行时间
        System.out.println("目标方法doSome()执行");
    }
}

MyAspect类

package com.hik;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
import java.util.Date;
/*
* @Aspect:是aspectJ这个框架中的注解。
* 作用:来表示当前类是切面类,切面类是给业务方法增加功能的类,在这个类中有切面的功能代码
* 注解的使用位置:在切面类的定义上面
* */
@Aspect
public class MyAspect {
    /*
    * 定义方法:方法是实现切面功能的。
    * 方法的定义要求:
    * 1.公共方法public
    * 2.方法没有返回值
    * 3.方法名称是自定义的
    * 4.方法可以有参数也可以没有参数。如果有参数,参数不是自定义的,有几个参数类型可供使用
    * */
    /*
    * @Before:前置通知注解
    * 这个注解是有属性值的,属性值是value:是切入点表达式execution(),表示切面的功能执行的位置
    *@Befor注解的位置:是在方法上面添加注解
    * 特点:
    * 1.在目标方法之前先执行
    * 2.不会改变目标方法的执行结果
    * 3.不会影响目标方法的运行
    * */
    /*execution(访问修饰符 包名.类名.方法名(参数类型1,参数类型2...))*/
    @Before(value = "execution(public void com.hik.SomeSeiveceImpl.doSome(String,Integer))")
    public void myBefore(){
        //就是切面要执行的功能代码
        System.out.println("前置通知,切面功能:在目标方法之前输出执行时间:" + new Date());
    }
}

main目录下的resource中设置配置文件:applicationcontext.xml

<?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: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/aop
       https://www.springframework.org/schema/aop/spring-aop.xsd">
    <!--把对象交给Spring容器,由Spring容器统一创建和管理对象-->
    <!--声明目标对象-->
    <bean id="someService" class="com.hik.SomeSeiveceImpl"></bean>
    <!--声明切面对象-->
    <bean id="myAspect" class="com.hik.MyAspect"></bean>
    <!--声明自动代理生成器:使用aspectJ框架内部的功能,创建目标对象的代理对象。
        创建代理对象是在内存中实现的,修改目标对象的内存中结构。
        创建为代理对象,所以,目标对象就是被修改后的代理对象-->
    <!--这个标签会把容器中所有的对象都找到,然后,按照切入点表达式的声明信息,找到目标生成目标的代理,
        会把Spring中的目标对象一次性都生成代理对象-->
    <aop:aspectj-autoproxy>
    </aop:aspectj-autoproxy>

测试类

package com.hik;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class MyTest {
    @Test
    public void Test01(){
        String config = "applicationcontext.xml";
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext(config);
        //从容器中获取目标对象
        SomeSeivece proxy = (SomeSeivece) applicationContext.getBean("someService");
        //通过代理对象执行方法,实现目标方法执行时,增强了功能
        proxy.doSome("Rita", 18);
    }
}

执行结果:

前置通知,切面功能:在目标方法之前输出执行时间:Thu Jan 13 16:30:16 CST 2022

目标方法doSome()执行

Process finished with exit code 0

相关文章
|
2天前
|
存储 安全 Java
Spring Boot 3 集成Spring AOP实现系统日志记录
本文介绍了如何在Spring Boot 3中集成Spring AOP实现系统日志记录功能。通过定义`SysLog`注解和配置相应的AOP切面,可以在方法执行前后自动记录日志信息,包括操作的开始时间、结束时间、请求参数、返回结果、异常信息等,并将这些信息保存到数据库中。此外,还使用了`ThreadLocal`变量来存储每个线程独立的日志数据,确保线程安全。文中还展示了项目实战中的部分代码片段,以及基于Spring Boot 3 + Vue 3构建的快速开发框架的简介与内置功能列表。此框架结合了当前主流技术栈,提供了用户管理、权限控制、接口文档自动生成等多项实用特性。
27 8
|
2月前
|
XML Java 数据安全/隐私保护
Spring Aop该如何使用
本文介绍了AOP(面向切面编程)的基本概念和术语,并通过具体业务场景演示了如何在Spring框架中使用Spring AOP。文章详细解释了切面、连接点、通知、切点等关键术语,并提供了完整的示例代码,帮助读者轻松理解和应用Spring AOP。
Spring Aop该如何使用
|
2月前
|
监控 安全 Java
什么是AOP?如何与Spring Boot一起使用?
什么是AOP?如何与Spring Boot一起使用?
81 5
|
2月前
|
Java 开发者 Spring
深入解析:Spring AOP的底层实现机制
在现代软件开发中,Spring框架的AOP(面向切面编程)功能因其能够有效分离横切关注点(如日志记录、事务管理等)而备受青睐。本文将深入探讨Spring AOP的底层原理,揭示其如何通过动态代理技术实现方法的增强。
81 8
|
2月前
|
Java 开发者 Spring
Spring AOP 底层原理技术分享
Spring AOP(面向切面编程)是Spring框架中一个强大的功能,它允许开发者在不修改业务逻辑代码的情况下,增加额外的功能,如日志记录、事务管理等。本文将深入探讨Spring AOP的底层原理,包括其核心概念、实现方式以及如何与Spring框架协同工作。
|
2月前
|
XML 监控 安全
深入调查研究Spring AOP
【11月更文挑战第15天】
50 5
|
2月前
|
Java 开发者 Spring
Spring AOP深度解析:探秘动态代理与增强逻辑
Spring框架中的AOP(Aspect-Oriented Programming,面向切面编程)功能为开发者提供了一种强大的工具,用以将横切关注点(如日志、事务管理等)与业务逻辑分离。本文将深入探讨Spring AOP的底层原理,包括动态代理机制和增强逻辑的实现。
51 4
|
3月前
|
人工智能 自然语言处理 前端开发
SpringBoot + 通义千问 + 自定义React组件:支持EventStream数据解析的技术实践
【10月更文挑战第7天】在现代Web开发中,集成多种技术栈以实现复杂的功能需求已成为常态。本文将详细介绍如何使用SpringBoot作为后端框架,结合阿里巴巴的通义千问(一个强大的自然语言处理服务),并通过自定义React组件来支持服务器发送事件(SSE, Server-Sent Events)的EventStream数据解析。这一组合不仅能够实现高效的实时通信,还能利用AI技术提升用户体验。
268 2
|
2天前
|
Java 测试技术 应用服务中间件
Spring Boot 如何测试打包部署
本文介绍了 Spring Boot 项目的开发、调试、打包及投产上线的全流程。主要内容包括: 1. **单元测试**:通过添加 `spring-boot-starter-test` 包,使用 `@RunWith(SpringRunner.class)` 和 `@SpringBootTest` 注解进行测试类开发。 2. **集成测试**:支持热部署,通过添加 `spring-boot-devtools` 实现代码修改后自动重启。 3. **投产上线**:提供两种部署方案,一是打包成 jar 包直接运行,二是打包成 war 包部署到 Tomcat 服务器。
24 10
|
16天前
|
Java 数据库连接 Maven
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)
自动装配是现在面试中常考的一道面试题。本文基于最新的 SpringBoot 3.3.3 版本的源码来分析自动装配的原理,并在文未说明了SpringBoot2和SpringBoot3的自动装配源码中区别,以及面试回答的拿分核心话术。
最新版 | 深入剖析SpringBoot3源码——分析自动装配原理(面试常考)