Spring | AOP思想(下)

简介: Spring | AOP思想(下)

前置通知@Before


8fd391a92e2b4e5db3320288be97225e.png

03524fb63fee456d89b1514155fb3d97.png

运行结果:


3ebe966bf4b94027aeb1f6a3ee275775.png


后置通知@After


和前置通知类似,只是在原始方法执行后执行通知,下面直接来看运行结果


27408cf1b8624864add5290826ff0318.png

运行结果:


bc4a4bc7f2454d448e997761c17acc76.png


环绕通知@Around

5941add7c0d741f2b4e1cf6766933c8f.png


运行结果:

453bc339baf0457d8ec7d799abe9545b.png


注意事项:


注意事项
1.环绕通知必须有ProceedingJoinPoint类型的方法形参,用于执行原始方法
2.如果没有这个形参,原始方法不执行
3.由于未知原始方法是否会抛异常,因此这个切面方法必须抛异常
4.如果原始方法有返回值,切面方法的返回值必须是Object类型
5.如果原始方法是void类型,切面方法也可以是Object类型

返回后通知@AfterReturning

ea16c7480e2a4e8a960ab5c0996a1680.png


运行结果:


e2f8e1fdff4e49269ace70e3b6622124.png


@After和@AfterReturning对比


@After @AfterReturning
原始方法发生异常时仍然通知 原始方法正常return才通知


抛出异常后通知@AfterThrowing

0340e723b63d4c4fb9ae853662d5a642.png

f22ad9153d64416498765f0a132861dc.png

运行结果:


0f6de3d5d62c4d01a32aaa8c77cf8a7c.png


6.案例:使用@Around统计业务层方法执行时间


1.准备工作


7c8ada343bae41c78a1ef09e0005037a.png


准备好了实体类,mapper接口和service和和实现类,以及三个配置文件和jdbc.properties文件,下面是通知类(使用Signature类获取执行的方法有关信息


@Component
@Aspect
public class Advice {
    // 为两个方法设置切入点
    @Pointcut("execution(* demo8.service.*Service.select*(..))")
    private void selectPt() {
    }
    // 设置切面
    @Around("selectPt()")
    public Object method(ProceedingJoinPoint point) throws Throwable {
        Signature signature = point.getSignature();
        String methodName = signature.getName(); // 获取方法名
        String typeName = signature.getDeclaringTypeName(); // 获取方法类型
        long startTime = System.currentTimeMillis(); // 获取执行前的系统时间
        Object o = null;
        for (int i = 0; i < 1000; i++) {
            o = point.proceed();
        }
        long endTime = System.currentTimeMillis(); // 获取执行后的系统时间
        System.out.println("执行"+typeName+"类型的"+methodName+"方法花费时间:" + (endTime - startTime) + "ms");
        return o;
    }
}

测试类:


@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(classes = SpringConfig.class)
public class UserServiceTest {
    @Autowired
    private UserService service;
    @Test
    public void testSelectAll() {
        List<User> users = service.selectAll();
        System.out.println(users);
        System.out.println(users.size());
        // 获取方法有关信息
    }
    @Test
    public void testSelectById() {
        User user = service.selectById(2);
        System.out.println(user);
    }
}

测试结果:

289fb786cf474135bc2d1d087135a549.png


7.AOP获取通知数据


获取方法参数


五种通知都可以获取方法参数,一般使用JoinPoint类型形参的getArgs()方法,在环绕通知中(@Around)使用ProceedJoinPoint类型的getArgs()方法。


举例:

(1)JoinPoint对象


@Before("fn1()")
    public void before(JoinPoint joinPoint) {
        Object[] args = joinPoint.getArgs(); // 1.前置通知获取切入点方法参数
        System.out.println(Arrays.toString(args));
        System.out.println("before ...");
    }

业务层接口和实现类:

c9a423ffbc0448dc87796decb619b7bd.png

e83b30b4c0d1479e9b566a9a9dbae2b7.png

主类和运行结果:

a6eab3bafb4f4808baf56162a6d9fdde.png

(2)ProceedJoinPoint对象


c054da58b756451bb7863cb904bfdf86.png


获取方法返回值

只有返回后通知@AfterReturning和环绕通知@Around可以获取

(1)环绕通知

4fcc1ca3f21f4f69961b18c0e3de3449.png

(2)返回后通知

使用Object类型的方法形参,并在注解上加returning = 形参名称

85fbb8b85a4c4a00b8150db3a644968d.png

获取异常信息

只有抛出异常后通知@AfterThrowing和环绕通知@Around可以获取

(1)环绕通知

首先在serviceImpl抛出一个异常

4ff56f572b394f4dab6313e638bcd0a8.png

02978eef7c21463d8f6b31c4a387cb37.png


(2)抛出异常后通知

使用Throwable 类型的形参,并在注解上加上throwing = 形参名称


a3c71ea4f3c74727ba126b69bac14e23.png


8.案例:使用@Around处理参数中的字符串

8695e1dd6a7946cab161c7e23de271e9.png


d3d1df4f95704d53a96b8c5117d7bb50.png

8673a7cf5866483dbb719a237701146a.png

9c35f57522584d61877793787c4c20f5.png

9.AOP总结:


这篇文章主要介绍了AOP面向切面编程的概念和作用,以及切入点、通知、切面的内容,后面介绍通知的类别(5类),最后介绍在通知中如何获取并处理数据。


三、结语


AOP是spring的核心思想之一,前面学习了IOC(控制反转),至此开始越来越了解spring的工作机制。

相关文章
|
1月前
|
XML Java 开发者
Spring Boot中的AOP实现
Spring AOP(面向切面编程)允许开发者在不修改原有业务逻辑的情况下增强功能,基于代理模式拦截和增强方法调用。Spring Boot通过集成Spring AOP和AspectJ简化了AOP的使用,只需添加依赖并定义切面类。关键概念包括切面、通知和切点。切面类使用`@Aspect`和`@Component`注解标注,通知定义切面行为,切点定义应用位置。Spring Boot自动检测并创建代理对象,支持JDK动态代理和CGLIB代理。通过源码分析可深入了解其实现细节,优化应用功能。
|
15天前
|
XML Java 测试技术
Spring AOP—通知类型 和 切入点表达式 万字详解(通俗易懂)
Spring 第五节 AOP——切入点表达式 万字详解!
76 25
|
15天前
|
XML 安全 Java
Spring AOP—深入动态代理 万字详解(通俗易懂)
Spring 第四节 AOP——动态代理 万字详解!
67 24
|
1月前
|
存储 安全 Java
Spring Boot 3 集成Spring AOP实现系统日志记录
本文介绍了如何在Spring Boot 3中集成Spring AOP实现系统日志记录功能。通过定义`SysLog`注解和配置相应的AOP切面,可以在方法执行前后自动记录日志信息,包括操作的开始时间、结束时间、请求参数、返回结果、异常信息等,并将这些信息保存到数据库中。此外,还使用了`ThreadLocal`变量来存储每个线程独立的日志数据,确保线程安全。文中还展示了项目实战中的部分代码片段,以及基于Spring Boot 3 + Vue 3构建的快速开发框架的简介与内置功能列表。此框架结合了当前主流技术栈,提供了用户管理、权限控制、接口文档自动生成等多项实用特性。
81 8
|
3月前
|
XML Java 数据安全/隐私保护
Spring Aop该如何使用
本文介绍了AOP(面向切面编程)的基本概念和术语,并通过具体业务场景演示了如何在Spring框架中使用Spring AOP。文章详细解释了切面、连接点、通知、切点等关键术语,并提供了完整的示例代码,帮助读者轻松理解和应用Spring AOP。
Spring Aop该如何使用
|
3月前
|
监控 安全 Java
什么是AOP?如何与Spring Boot一起使用?
什么是AOP?如何与Spring Boot一起使用?
111 5
|
3月前
|
Java 开发者 Spring
深入解析:Spring AOP的底层实现机制
在现代软件开发中,Spring框架的AOP(面向切面编程)功能因其能够有效分离横切关注点(如日志记录、事务管理等)而备受青睐。本文将深入探讨Spring AOP的底层原理,揭示其如何通过动态代理技术实现方法的增强。
103 8
|
3月前
|
Java 开发者 Spring
Spring AOP 底层原理技术分享
Spring AOP(面向切面编程)是Spring框架中一个强大的功能,它允许开发者在不修改业务逻辑代码的情况下,增加额外的功能,如日志记录、事务管理等。本文将深入探讨Spring AOP的底层原理,包括其核心概念、实现方式以及如何与Spring框架协同工作。
|
3月前
|
XML 监控 安全
深入调查研究Spring AOP
【11月更文挑战第15天】
62 5
|
3月前
|
Java 开发者 Spring
Spring AOP深度解析:探秘动态代理与增强逻辑
Spring框架中的AOP(Aspect-Oriented Programming,面向切面编程)功能为开发者提供了一种强大的工具,用以将横切关注点(如日志、事务管理等)与业务逻辑分离。本文将深入探讨Spring AOP的底层原理,包括动态代理机制和增强逻辑的实现。
72 4