首页 > 编程语言 >Java日志记录几种实现方案

Java日志记录几种实现方案

时间:2024-12-07 16:31:47浏览次数:3  
标签:info Java log 几种 报错 日志 执行 监听

在平时使用到一些软件中,比如某宝或者某书,通过记录用户的行为来构建和分析用户的行为数据,同时也能更好优化产品设计和提升用户体验。比如在一个订单系统中,需要确定追踪用户的行为,比如:

  • 登录/登出
  • 浏览商品
  • 加购商品
  • 搜索商品关键字
  • 下单

上述行为就需要使用到日志系统来存储或者记录数据,Java 有几种日志方案,简单介绍几种实现方案,以及需要注意的点。

日志添加需要注意问题

根据业务的不同,需要使用匹配到合适的日志方案。也需要注意几个问题:

  • 不能影响原来的业务逻辑。
  • 不能报错,即使报错,不能影响原有的业务代码。
  • 对于耗时的日志代码,使用异步方法
  • 侵入性低,尽量少改动原始代码

Spring AOP

Spring AOP 通过切面编程实现不修改原有代码,而动态添加功能的能力。这种方式有以下几个好处:

  • 侵入性低
  • 可重用性强

本文使用基于注解的 AOP,首先定义一个切面注解 AopTest:

/**
 * 切面注解
 */
@Target({ElementType.TYPE, ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface AopTest {
}

再创建通知 @Around :

@Around("@annotation(com.test.annotation.AopTest)")
public Object annotationTest(ProceedingJoinPoint joinPoint) throws Throwable {
    log.info("执行前");
    Object result = joinPoint.proceed(); // 执行目标方法
    Object[] args = joinPoint.getArgs();
    log.info("执行后");
    return result;
}

一般都会在执行后添加日志即可,想要在那个方法或者接口加日志,只需要在方法上添加注解即可,比如在接口添加注解:

@GetMapping
@AopTest
public String first(String param) {
    log.info("执行first方法");
    return "result " + param;
}

请求接口后,就有如下的输出:

执行前
执行first方法
执行后

但是切面有一个问题,执行切面报错,方法也无法执行,就需要捕获异常,保证业务代码正常执行,改造一下上面的通知:

@Around("@annotation(com.test.annotation.AopTest)")
public Object annotationTest(ProceedingJoinPoint joinPoint) throws Throwable {log.info("执行前");
    log.info("执行前");
    Object result = joinPoint.proceed(); // 执行目标方法
    try {
        Object[] args = joinPoint.getArgs();
        // 添加日志
        log.info("执行后");
        int a = 1/0;
    } catch (Exception e) {
        log.error(e.getMessage());
    }
    return result;
}

改造后,即使切面报错,也不会影响业务代码的执行了。

AOP 是同步执行的,如果日志添加是一个比较耗时的操作,也会影响接口的响应速度,此时可以使用异步的方式,比如消息队列。

总结一下,Spring AOP 有以下几个优点和缺点。

  • 优点:

    • 侵入性低
    • 可重用性强

缺点及解决方案:

1、切面报错可能会影响业务代码

  • 问题:在切面的异常如果没有正确处理,可能会影响业务代码的正常执行。
  • 解决方案:
    • 捕获异常,确保业务代码正常执行
    • 一般使用try catch 捕获异常,防止向上传播

2、同步执行会影响接口响应速度

  • 问题:如果切面中存在耗时的操作,同步操作会导致接口的响应速度变慢。
  • 解决方案:
    • 通过消息队列将耗时操作异步执行
    • 使用 @Async 将方法异步执行,将任务从主线程脱离出来,交给其他线程池执行

事件监听 + 异步

Spring 事件监听机制是一种发布-订阅模式,将事件的发布和监听解耦开来。通过这种机制,事件的发布者无需关注监听的逻辑,监听者也无需直接依赖发布者。

Spring 事件监听有三个部分组成:

  1. 事件

自定义一个事件,继承 ApplicationEvent:

@Getter
@Setter
public class DemoEvent extends ApplicationEvent {

    private String name;

    public DemoEvent(Object source) {
        super(source);
    }
}

  1. 事件监听

基于 @EventListener 注解监听事件:

@Component
@Slf4j
public class DemoEventListener {

  @EventListener
  public void handleEvent(DemoEvent event){
      log.info(event.getName());
      log.info("事件监听");

  }
}

  1. 事件发布

使用 applicationEventPublisher.publishEvent 方法发布事件,


@Autowired
private ApplicationEventPublisher applicationEventPublisher;

@Override
public void publish() {
    // 执行业务代码
    log.info("执行登录,当前时间 {}",LocalTime.now());
    DemoEvent event = new DemoEvent(this);
    event.setName("hello");
    applicationEventPublisher.publishEvent(event);
    log.info("完成登录,当前时间 {}",LocalTime.now());
    log.info("执行 service");
}

配置好事件的发布和监听之后,在业务代码添加事件的发布,监听方法内添加日志的记录。

Spring 事件监听虽然解耦了发布和监听,只是解耦逻辑代码,两者还是同步执行,并且都是在同一个线程执行的,所以事件监听也无法解决添加日志报错,以及耗时的问题。

  1. 验证监听方法是否同步

在事件监听添加延迟:

  @EventListener
  public void handleEvent(DemoEvent event){
      try {
          Thread.sleep(2000);
      } catch (InterruptedException e) {
          e.printStackTrace();
      }
      log.info(event.getName());
      log.info("事件监听");

  }

控制台输出:

执行登录,当前时间 16:52:30.799
hello
事件监听
完成登录,当前时间 16:52:33.799
执行 service

接口执行了 3 秒多,并且 publish 方法要等待监听方法执行完毕之后才能执行,说明事件监听是同步的

  1. 验证监听方法报错是否会影响主流程

在监听方法添加错误代码:

@EventListener
public void handleEvent(DemoEvent event){
    int a = 1/0;
    log.info(event.getName());
    log.info("事件监听");

}

控制台输出:

执行登录,当前时间 17:10:08.396
java.lang.ArithmeticException: / by zero

监听方法报错,接口也报错,业务代码无法执行,说明监听方法报错会影响事件发布方法

解决报错的问题,使用异常捕获即可。而延迟的问题,就需要使用到异步的操作,异步就是另启一个线程执行监听方法,异步除了能解决延迟的问题,也顺便解决了报错的问题。

实现异步在监听方法上添加 @Async 异步注解,监听方法添加延迟和错误代码:

执行登录,当前时间 17:21:50.971
完成登录,当前时间 17:21:50.974
执行 service

publish 方法既不会延迟,也不会因为监听报错影响执行,异步完美解决耗时和报错的问题

消息队列

海量日志场景,消息队列是一个很好的选择,它也是解耦了发布者和订阅者,如果订阅者开启了手动确认,消费者也需要使用 try catch 捕获异常信息,确保消息能正常消费。

总结

本文介绍几种日志记录的实现方案:

  • Spring AOP:
    • 优点: 侵入性低,代码可重复性强
    • 问题以及解决方案:
      • 切面中可能会报错,报错会影响业务代码的正常执行,解决方法是使用 try catch 捕获异常。
      • 日志记录会影响业务代码执行效率,可以使用消息队列异步执行日志操作
  • 事件监听 + 异步
    • 优点: 解耦业务逻辑和日志记录,提升代码的内聚性。
    • 缺点以及解决方案:
      • AOP 存在的问题,事件监听同样存在,报错和耗时都会影响业务代码。报错可以使用异常捕获,延迟问题可以使用异步方式解决,而异步另起线程也顺便解决了报错影响业务代码的问题。
  • 消息队列
    • 优点: 使用于高并发日记记录场景
    • 问题: 增加系统的复杂性和稳定性,还需要考虑消息的丢失和重复消费问题。

标签:info,Java,log,几种,报错,日志,执行,监听
From: https://www.cnblogs.com/jeremylai7/p/18592368

相关文章

  • Java春招大厂面试题
    文章目录软件开发2024春招大厂面试题1.redis的跳表是怎么实现的?2.MVCC实现原理3.Redis怎么实现延时消息4.http默认端口号是多少5.http的几种请求(get、post…)有什么区别1.GET2.POST6.Http和Https的区别7.tcp协议中,三次握手四次挥手是怎样的1.什么是三次握手2.......
  • 蓝桥杯 2024 省赛 C++ B组 R 格式 (JAVA面向对象 高精度 纯api题解)
    解题思路:由于数位较大这里采用高精度,又因为高精度写起来比较麻烦所以这里直接采用JAVAapi中的高精度浮点数类型和高精度整数类型,应为高精度浮点数类型四舍五入较为麻烦所以这里改为手动四舍五入importjava.math.BigDecimal;importjava.math.BigInteger;importjava.util......
  • 接龙数列(第十四届蓝桥杯C++B组JAVA题解 动态规划)
    对于一个长度为 KK 的整数数列:A1,A2,...,AK,我们称之为接龙数列当且仅当 Ai 的首位数字恰好等于 Ai−1 的末位数字(2≤i≤K)。例如 12,23,35,56,61,11是接龙数列;12,23,34,56不是接龙数列,因为 56的首位数字不等于 3434 的末位数字。所有长度为 11 的整数数列都......
  • 网上图书购物管理系统|Java|SSM|VUE| 前后端分离
    【一】可以提供远程部署安装,包扩环境【二】提供软件相关的安装包【三】如果需要提供java入门资料可咨询             【技术栈】1⃣️:架构:B/S、MVC2⃣️:系统环境:Windowsh/Mac3⃣️:开发环境:IDEA、JDK1.8、Maven、Mysql5.7+4⃣️:技术栈:Java、Mysql、S......
  • 基于微信平台健身小助手小程序的设计与实现【java或python】-计算机毕业设计源码+LW文
    选题的意义与目的网络和科技的进步以及人们生活条件的提高都让微信小程序越来越平民化,深入日常生活中。我国非常强调体育以及健身,需要不断的让更多人参与到健身中,因为健身不仅可以锻炼身体,也能锻炼意志,有了健康的身体,就可以好好的努力工作,努力学习,为国家做出应有的贡献。有一......
  • 网上图书购物管理系统|Java|SSM|VUE| 前后端分离
    【一】可以提供远程部署安装,包扩环境【二】提供软件相关的安装包【三】如果需要提供java入门资料可咨询             【技术栈】1⃣️:架构:B/S、MVC2⃣️:系统环境:Windowsh/Mac3⃣️:开发环境:IDEA、JDK1.8、Maven、Mysql5.7+4⃣️:技术栈:Java、Mysql、S......
  • 面试题:JavaScript+ES5+
    jsthis指向看函数的调用方式,而不是他的定义时候分类构造函数==>new时候创建的对象对象的方法内部==》调用方法的对象事件处理函数==》绑定的事件箭头函数==》没有自己的this其他函数(全局的/局部的)==》匿名的就是window定时器函数==》window立即执行函数==》w......
  • 介绍一下 WebApplicationContext 思维导图 代码示例(java 架构)
    WebApplicationContext是Spring框架中的一个接口,它是ApplicationContext的扩展,专门用于Web应用程序。它提供了对Web特定功能的支持,例如解析主题(themes)、管理国际化资源、以及与Servlet容器集成等。下面是一个关于WebApplicationContext的思维导图大纲和一些代码示例。WebAp......
  • gorm: 配置日志输出到文件
    一,官方文档地址:https://gorm.io/zh_CN/docs/logger.html二,代码例子:1,写日志到文件的函数://日志写入到文件funcLogToFile(filename,msgstring){ //输出到文件 //filename:="logs/"+time.Now().Format("2006-01-02")+".log" file,err:=os.OpenFile(file......
  • Java项目:小徐影城管理系统(java+SpringBoot+Mybaits+Vue+elementui+mysql)
    源码获取:俺的博客首页"资源"里下载! 项目介绍Springboot+vue小徐影城管理系统环境需要1.运行环境:最好是javajdk1.8,我们在这个平台上运行的。其他版本理论上也可以。2.IDE环境:IDEA,Eclipse,Myeclipse都可以。推荐IDEA;3.硬件环境:windows7/8/101G内存以上;或者MacO......