通过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月前
|
XML Java 数据安全/隐私保护
Spring Aop该如何使用
本文介绍了AOP(面向切面编程)的基本概念和术语,并通过具体业务场景演示了如何在Spring框架中使用Spring AOP。文章详细解释了切面、连接点、通知、切点等关键术语,并提供了完整的示例代码,帮助读者轻松理解和应用Spring AOP。
Spring Aop该如何使用
|
2月前
|
监控 安全 Java
什么是AOP?如何与Spring Boot一起使用?
什么是AOP?如何与Spring Boot一起使用?
68 5
|
2月前
|
Java 开发者 Spring
深入解析:Spring AOP的底层实现机制
在现代软件开发中,Spring框架的AOP(面向切面编程)功能因其能够有效分离横切关注点(如日志记录、事务管理等)而备受青睐。本文将深入探讨Spring AOP的底层原理,揭示其如何通过动态代理技术实现方法的增强。
76 8
|
2月前
|
Java 开发者 Spring
Spring AOP 底层原理技术分享
Spring AOP(面向切面编程)是Spring框架中一个强大的功能,它允许开发者在不修改业务逻辑代码的情况下,增加额外的功能,如日志记录、事务管理等。本文将深入探讨Spring AOP的底层原理,包括其核心概念、实现方式以及如何与Spring框架协同工作。
|
2月前
|
XML 监控 安全
深入调查研究Spring AOP
【11月更文挑战第15天】
49 5
|
2月前
|
Java 开发者 Spring
Spring AOP深度解析:探秘动态代理与增强逻辑
Spring框架中的AOP(Aspect-Oriented Programming,面向切面编程)功能为开发者提供了一种强大的工具,用以将横切关注点(如日志、事务管理等)与业务逻辑分离。本文将深入探讨Spring AOP的底层原理,包括动态代理机制和增强逻辑的实现。
48 4
|
3月前
|
存储 缓存 Java
Spring高手之路23——AOP触发机制与代理逻辑的执行
本篇文章深入解析了Spring AOP代理的触发机制和执行流程,从源码角度详细讲解了Bean如何被AOP代理,包括代理对象的创建、配置与执行逻辑,帮助读者全面掌握Spring AOP的核心技术。
56 3
Spring高手之路23——AOP触发机制与代理逻辑的执行
|
2月前
|
Java Spring
[Spring]aop的配置与使用
本文介绍了AOP(面向切面编程)的基本概念和核心思想。AOP是Spring框架的核心功能之一,通过动态代理在不修改原代码的情况下注入新功能。文章详细解释了连接点、切入点、通知、切面等关键概念,并列举了前置通知、后置通知、最终通知、异常通知和环绕通知五种通知类型。
43 1
|
4月前
|
设计模式 Java 测试技术
spring复习04,静态代理动态代理,AOP
这篇文章讲解了Java代理模式的相关知识,包括静态代理和动态代理(JDK动态代理和CGLIB),以及AOP(面向切面编程)的概念和在Spring框架中的应用。文章还提供了详细的示例代码,演示了如何使用Spring AOP进行方法增强和代理对象的创建。
spring复习04,静态代理动态代理,AOP
|
2月前
|
安全 Java 测试技术
Java开发必读,谈谈对Spring IOC与AOP的理解
Spring的IOC和AOP机制通过依赖注入和横切关注点的分离,大大提高了代码的模块化和可维护性。IOC使得对象的创建和管理变得灵活可控,降低了对象之间的耦合度;AOP则通过动态代理机制实现了横切关注点的集中管理,减少了重复代码。理解和掌握这两个核心概念,是高效使用Spring框架的关键。希望本文对你深入理解Spring的IOC和AOP有所帮助。
45 0