通过Spring AOP来深入理解代理模式

本文涉及的产品
日志服务 SLS,月写入数据量 50GB 1个月
简介: 1. 背景描述通常我们在项目中,都需要打印日志,方便系统的维护和排查错误的需要,如下所示。

1. 背景描述

通常我们在项目中,都需要打印日志,方便系统的维护和排查错误的需要,如下所示。

@Service
public class PredictService {
    private static final Logger LOGGER = LoggerFactory.getLogger(PredictController.class);
    public Response intent(IntentPredictVO vo) {
        LOGGER.info("意图预测入口,robotId:{},questionId:{}, question:{}",
                vo.getRobotId(),vo.getQuestionId(), vo.getQuestion());
        Response<IntentPredictDTO> dto = ...;
        LOGGER.info("意图预测返回,robotId:{},questionId:{}, question:{},结果:{}",
                vo.getQuestionId(), vo.getQuestion(), JSONObject.toJSONString(dto));
        return Response.success(dto);
    }
    public Response nextintents(NextIntentPredictVO vo) {
        LOGGER.info("下一个意图预测入口,robotId:{},questionId:{}, question:{}",
                vo.getRobotId(),vo.getQuestionId(), vo.getQuestion());
        Response<NextIntentPredictDTO> process = ...;
        LOGGER.info("下一个意图预测返回,robotId:{},questionId:{}, question:{},结果:{}",
                vo.getQuestionId(), vo.getQuestion(), JSONObject.toJSONString(process.getData()));
        return process;
    }
    ...
}
复制代码

从上面的代码,我们发现一个问题,其实基本都是重复的日志代码,让系统看着特别臃肿,接下来我们就讲解如何通过Spring AOP(代理模式-动态代理)来改善上面的情况,让系统更加简洁。

什么是代理模式?

通过代理控制对对象的访问,可以详细控制访问某个(某类)对象的方法,在调用这个方法之前做前置处理,调用这个方法后做后置处理。

真实角色--定义真实角色所要实现的业务逻辑,供代理角色调用。关注真正的业务逻辑。(相当于明星)

代理角色--是真实角色的代理,通过真实角色的业务逻辑方法实现抽象方法,并可以附加自己的操作。将统一的流程控制放到代理角色中处理。(相当于经纪人)

下面我们就用Spring AOP来改造日志,进行统一处理。


2. 编写拦截规则的注解

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface PredictLog {
  String value() default "";
}
复制代码


3. 编写使用注解的被拦截类(真实角色)

@Service
public class PredictService{
    @PredictLog("意图预测")
    public Response intent(IntentPredictVO vo) {
        Response<IntentPredictDTO> dto = ...;
        return Response.success(dto);
    }
    @PredictLog("下一个意图预测")
    public Response nextintents(NextIntentPredictVO vo) {
        Response<NextIntentPredictDTO> process = ...;
        return process;
    }
    ...
}
复制代码


4. 编写切面-动态代理(代理角色:核心方法做统一的流程控制)

注意:

  • 上面的NextIntentPredictVO 和IntentPredictVO 都是继承RobotQuestionVO。
  • 这里简单理解成代理角色(Spring内部使用CGLIB代理或JDK动态代理动态生成代理类和对象)
@Aspect
@Component
@Slf4j
public class PredictLogAop {
  @Pointcut("@annotation(predictLog)")
  public void predictLogCut(PredictLog predictLog) {
  }
  @Around("predictLogCut(predictLog)")
  public Object predictLogCutAround(ProceedingJoinPoint pjp, PredictLog predictLog) throws Throwable {
    String businessName = predictLog.value();
    RobotQuestionVO robotQuestion = null;
    Object[] args = pjp.getArgs();
    if (args.length > 0) {
      String params = JSONObject.toJSONString(args[0]);
      robotQuestion = JSONObject.parseObject(params, RobotQuestionVO.class);
      log.info("{}入口, 请求参数: {} .", businessName, JSONObject.toJSONString(robotQuestion));
    }
    // 执行方法
    Object result = pjp.proceed();
    log.info("{}返回, 请求参数: {},结果: {} .", businessName, JSONObject.toJSONString(robotQuestion),
        JSONObject.toJSONString(result));
    return result;
  }
}
复制代码


5. 运行结果

public class main {
    public static void main(String[] args) {
        AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AopApplication.class);
        PredictService predictService = context.getBean(PredictService.class);
        predictService.intent();
        predictService.nextintents();
    }
}
复制代码

结果如下:

意图预测入口, 请求参数: {"question":"我的贷款还未审核","questionId":"C875FFC0C66B2DF1D725A0BCC905EA28_2","robotId":"xxxx"} .

...(真正的业务逻辑)

意图预测返回, 请求参数: {"question":"我的贷款还未审核","questionId":"C875FFC0C66B2DF1D725A0BCC905EA28_2","robotId":"xxxx"}, 结果: {"code":"10000","message":"ok"} .

下一个意图预测入口, 请求参数: {"question":"我的贷款还未审核","questionId":"C875FFC0C66B2DF1D725A0BCC905EA28_2","robotId":"xxxx"} .

...(真正的业务逻辑)

下一个意图预测返回, 请求参数: {"question":"我的贷款还未审核","questionId":"C875FFC0C66B2DF1D725A0BCC905EA28_2","robotId":"xxxx"}, 结果: {"code":"10000","message":"ok"} .


6. 总结

代理模式给我们带来的好处:

  • 可以使真实角色操作更加纯粹, 不用去关心一些闲杂的事情。
  • 闲杂事情让代理去做,实现业务的分工。
  • 当闲杂事情发生变化的时候, 真实角色是不用去管的, 代理角色去管就行, 这样就方便各角色的集中管理。

本文通过Spring AOP将日志代码抽离,基于注解实现打印重复日志,简化了系统。


相关实践学习
日志服务之使用Nginx模式采集日志
本文介绍如何通过日志服务控制台创建Nginx模式的Logtail配置快速采集Nginx日志并进行多维度分析。
相关文章
|
2月前
|
Java
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
这篇文章是Spring5框架的实战教程,深入讲解了AOP的基本概念、如何利用动态代理实现AOP,特别是通过JDK动态代理机制在不修改源代码的情况下为业务逻辑添加新功能,降低代码耦合度,并通过具体代码示例演示了JDK动态代理的实现过程。
Spring5入门到实战------9、AOP基本概念、底层原理、JDK动态代理实现
|
13天前
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP
|
3月前
|
Java Spring
在Spring Boot中使用AOP实现日志切面
在Spring Boot中使用AOP实现日志切面
|
4天前
|
Java Spring 容器
Spring IOC、AOP与事务管理底层原理及源码解析
Spring框架以其强大的控制反转(IOC)和面向切面编程(AOP)功能,成为Java企业级开发中的首选框架。本文将深入探讨Spring IOC和AOP的底层原理,并通过源码解析来揭示其实现机制。同时,我们还将探讨Spring事务管理的核心原理,并给出相应的源码示例。
30 9
|
26天前
|
Java 数据库连接 数据库
Spring基础3——AOP,事务管理
AOP简介、入门案例、工作流程、切入点表达式、环绕通知、通知获取参数或返回值或异常、事务管理
Spring基础3——AOP,事务管理
|
2月前
|
XML Java 数据格式
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
这篇文章是Spring5框架的AOP切面编程教程,通过XML配置方式,详细讲解了如何创建被增强类和增强类,如何在Spring配置文件中定义切入点和切面,以及如何将增强逻辑应用到具体方法上。文章通过具体的代码示例和测试结果,展示了使用XML配置实现AOP的过程,并强调了虽然注解开发更为便捷,但掌握XML配置也是非常重要的。
Spring5入门到实战------11、使用XML方式实现AOP切面编程。具体代码+讲解
|
2月前
|
缓存 Java 开发者
Spring高手之路22——AOP切面类的封装与解析
本篇文章深入解析了Spring AOP的工作机制,包括Advisor和TargetSource的构建与作用。通过详尽的源码分析和实际案例,帮助开发者全面理解AOP的核心技术,提升在实际项目中的应用能力。
26 0
Spring高手之路22——AOP切面类的封装与解析
|
2月前
|
安全 Java 开发者
Java 新手入门:Spring 两大利器IoC 和 AOP,小白也能轻松理解!
Java 新手入门:Spring 两大利器IoC 和 AOP,小白也能轻松理解!
31 1
|
2月前
|
Java Spring
Spring的AOP组件详解
该文章主要介绍了Spring AOP(面向切面编程)组件的实现原理,包括Spring AOP的基础概念、动态代理模式、AOP组件的实现以及Spring选择JDK动态代理或CGLIB动态代理的依据。
Spring的AOP组件详解
|
2月前
|
Java API Spring
Spring Boot 中的 AOP 处理
对 Spring Boot 中的切面 AOP 做了详细的讲解,主要介绍了 Spring Boot 中 AOP 的引入,常用注解的使用,参数的使用,以及常用 api 的介绍。AOP 在实际项目中很有用,对切面方法执行前后都可以根据具体的业务,做相应的预处理或者增强处理,同时也可以用作异常捕获处理,可以根据具体业务场景,合理去使用 AOP。