深入理解Spring IOC之扩展篇(六)、基于Aop的自定义注解

简介: 深入理解Spring IOC之扩展篇(六)、基于Aop的自定义注解

上篇我们基于Spring的各种组件和注解把我们的业务逻辑和Spring进行了集成,其中我们定义了很多我们自己的注解。在本篇我们也将定义我们自己的注解,但是目的和上篇不同的是,上篇目的自定义注解是为了集成,而这篇的自定义注解是为了增强。

在demo开始之前,我先简单说下Aop中的这几个重要概念:

连接点(Joinpoint):在程序执行过程中某个特定的点,比如类初始化前、类初始化后,方法调用前,方法调用后;

切点(Pointcut):所谓切点就是你所切取的类中的方法,比如你横切的这个类中有两个方法,那么这两个方法都是连接点,对这两个方法的定位就称之为切点;

增强(Advice):增强是织入到连接点上的一段程序,另外它还拥有连接点的相关信息;

目标对象(Target):增强逻辑的织入目标类,就是我的增强逻辑植入到什么位置;

引介(Introduction):一种特殊的增强,它可以为类添加一些属性喝方法;

织入(Weaving):织入就是讲增强逻辑添加到目标对象的过程;

代理(Proxy):一个类被AOP织入增强后,就会产生一个结果类,他是融合了原类和增强逻辑的代理类;

切面(Aspect):切面由切点和增强组成,他是横切逻辑定义和连接点定义的组成;

我们来看看具体的做法:

首先按照惯例,我们需要有我们自己的注解


@Retention(RetentionPolicy.RUNTIME)
@Target({ElementType.METHOD})
public @interface MyAspect {
}


然后是自己注解的处理类:


@Aspect
// 注意这里必须有Component注解
@Component
public class AspectHandler {
    @Pointcut(value = "@annotation(com.example.demo.external6.MyAspect)")
    public void myJoinPoint(){}
    @Around(value = "myJoinPoint()")
    public Object around(ProceedingJoinPoint joinPoint) throws Throwable {
        System.out.println("around before....");
        joinPoint.proceed();
        System.out.println("around after....");
        return new MyEntity();
    }
    @Before(value = "myJoinPoint()", argNames = "joinPoint")
    public void before(JoinPoint joinPoint){
        System.out.println("before ======");
        MyEntity entity = (MyEntity)joinPoint.getArgs()[0];
        System.out.println(entity.getId());
        System.out.println("before ======");
    }
    @After(value = "myJoinPoint()", argNames = "joinPoint")
    public void after(JoinPoint joinPoint){
        System.out.println("after ==========");
        MyEntity entity = (MyEntity)joinPoint.getArgs()[0];
        System.out.println(entity.getId());
        System.out.println("after ==========");
    }
    @AfterReturning(value = "myJoinPoint()", argNames = "joinPoint,r", returning = "r")
    public void afterReturning(JoinPoint joinPoint,Object r){
        System.out.println("afterReturning");
        MyEntity r1 = (MyEntity)r;
        System.out.println(r1.getId());
    }
    @AfterThrowing(value = "myJoinPoint()", argNames = "joinPoint,tx",throwing = "tx")
    public void afterThrowing(JoinPoint joinPoint,Throwable tx){
        System.out.println("afterThrowing");
        System.out.println(tx.getMessage());
    }
}


用作测试的业务代码


@Component
public class TestService{
    @MyAspect
    public MyEntity doService(MyEntity entity){
//        if(entity.getId() == 1){
//            throw new RuntimeException("挂了");
//        }
        entity.setId(2);
        System.out.println("do service");
        return new MyEntity();
    }
}


简单构造一个实体类:


@Data
public class MyEntity {
    private int id;
}


配置类


@Configuration
@ComponentScan(basePackages = "com.example.demo.external6")
@EnableAspectJAutoProxy
public class Config {
}


注意EnableAspectJAutoProxy其中一个参数是expose-proxy,默认为false,设置为true之后你可以在被代理的类中使用AopUtil.currentProxy()方法来获取当前类的代理。另外一个参数proxyTargetClass是为你的类设置什么样的代理方式,我们知道spring aop当你的类实现了接口的时候,它会为你使用jdk动态代理,而当没有实现接口的时候会使用cglib代理,当我们把proxyTargetClass设置为true的时候,不论什么时候,都会使用cglib代理。

测试代码:


public static void main(String[] args) {
        AnnotationConfigApplicationContext annotationConfigApplicationContext = new AnnotationConfigApplicationContext(Config.class);
        TestService testService = (TestService) annotationConfigApplicationContext.getBean("testService");
        MyEntity a = new MyEntity();
        a.setId(1);
        testService.doService(a);
    }


输出结果:


1686813573805.png


我们其实可以轻松看到执行顺序。然后我们再测试测试异常的结果,测试代码其实就是把TestService中的异常放开就行。

测试结果:


1686813581381.png


两个测试结果均符合我们的预期。

目录
相关文章
|
2月前
|
缓存 监控 Java
SpringBoot @Scheduled 注解详解
使用`@Scheduled`注解实现方法周期性执行,支持固定间隔、延迟或Cron表达式触发,基于Spring Task,适用于日志清理、数据同步等定时任务场景。需启用`@EnableScheduling`,注意线程阻塞与分布式重复问题,推荐结合`@Async`异步处理,提升任务调度效率。
462 128
|
2月前
|
XML 安全 Java
使用 Spring 的 @Aspect 和 @Pointcut 注解简化面向方面的编程 (AOP)
面向方面编程(AOP)通过分离横切关注点,如日志、安全和事务,提升代码模块化与可维护性。Spring 提供了对 AOP 的强大支持,核心注解 `@Aspect` 和 `@Pointcut` 使得定义切面与切入点变得简洁直观。`@Aspect` 标记切面类,集中处理通用逻辑;`@Pointcut` 则通过表达式定义通知的应用位置,提高代码可读性与复用性。二者结合,使开发者能清晰划分业务逻辑与辅助功能,简化维护并提升系统灵活性。Spring AOP 借助代理机制实现运行时织入,与 Spring 容器无缝集成,支持依赖注入与声明式配置,是构建清晰、高内聚应用的理想选择。
383 0
|
1月前
|
XML Java 应用服务中间件
【SpringBoot(一)】Spring的认知、容器功能讲解与自动装配原理的入门,带你熟悉Springboot中基本的注解使用
SpringBoot专栏开篇第一章,讲述认识SpringBoot、Bean容器功能的讲解、自动装配原理的入门,还有其他常用的Springboot注解!如果想要了解SpringBoot,那么就进来看看吧!
320 2
|
2月前
|
XML Java 数据格式
常用SpringBoot注解汇总与用法说明
这些注解的使用和组合是Spring Boot快速开发和微服务实现的基础,通过它们,可以有效地指导Spring容器进行类发现、自动装配、配置、代理和管理等核心功能。开发者应当根据项目实际需求,运用这些注解来优化代码结构和服务逻辑。
277 12
|
2月前
|
Java 测试技术 数据库
使用Spring的@Retryable注解进行自动重试
在现代软件开发中,容错性和弹性至关重要。Spring框架提供的`@Retryable`注解为处理瞬时故障提供了一种声明式、可配置的重试机制,使开发者能够以简洁的方式增强应用的自我恢复能力。本文深入解析了`@Retryable`的使用方法及其参数配置,并结合`@Recover`实现失败回退策略,帮助构建更健壮、可靠的应用程序。
279 1
使用Spring的@Retryable注解进行自动重试
|
2月前
|
监控 安全 Java
使用 @HealthEndpoint 在 Spring Boot 中实现自定义健康检查
Spring Boot 通过 Actuator 模块提供了强大的健康检查功能,帮助开发者快速了解应用程序的运行状态。默认健康检查可检测数据库连接、依赖服务、资源可用性等,但在实际应用中,业务需求和依赖关系各不相同,因此需要实现自定义健康检查来更精确地监控关键组件。本文介绍了如何使用 @HealthEndpoint 注解及实现 HealthIndicator 接口来扩展 Spring Boot 的健康检查功能,从而提升系统的可观测性与稳定性。
197 0
使用 @HealthEndpoint 在 Spring Boot 中实现自定义健康检查
|
2月前
|
传感器 Java 数据库
探索Spring Boot的@Conditional注解的上下文配置
Spring Boot 的 `@Conditional` 注解可根据不同条件动态控制 Bean 的加载,提升应用的灵活性与可配置性。本文深入解析其用法与优势,并结合实例展示如何通过自定义条件类实现环境适配的智能配置。
166 0
探索Spring Boot的@Conditional注解的上下文配置
|
2月前
|
智能设计 Java 测试技术
Spring中最大化@Lazy注解,实现资源高效利用
本文深入探讨了 Spring 框架中的 `@Lazy` 注解,介绍了其在资源管理和性能优化中的作用。通过延迟初始化 Bean,`@Lazy` 可显著提升应用启动速度,合理利用系统资源,并增强对 Bean 生命周期的控制。文章还分析了 `@Lazy` 的工作机制、使用场景、最佳实践以及常见陷阱与解决方案,帮助开发者更高效地构建可扩展、高性能的 Spring 应用程序。
115 0
Spring中最大化@Lazy注解,实现资源高效利用
|
2月前
|
安全 IDE Java
Spring 的@FieldDefaults和@Data:Lombok 注解以实现更简洁的代码
本文介绍了如何在 Spring 应用程序中使用 Project Lombok 的 `@Data` 和 `@FieldDefaults` 注解来减少样板代码,提升代码可读性和可维护性,并探讨了其适用场景与限制。
122 0
Spring 的@FieldDefaults和@Data:Lombok 注解以实现更简洁的代码
|
2月前
|
Java 测试技术 编译器
@GrpcService使用注解在 Spring Boot 中开始使用 gRPC
本文介绍了如何在Spring Boot应用中集成gRPC框架,使用`@GrpcService`注解实现高效、可扩展的服务间通信。内容涵盖gRPC与Protocol Buffers的原理、环境配置、服务定义与实现、测试方法等,帮助开发者快速构建高性能的微服务系统。
488 0