最近线上排查问题时候,发现请求太多导致日志错综复杂,没办法把用户在一次或多次请求的日志关联在一起,所以就利用SpringBoot+Logback手写了一个简单的链路追踪,下面详细介绍下。

一、实现原理

Spring Boot默认使用LogBack日志系统,并且已经引入了相关的jar包,所以我们无需任何配置便可以使用LogBack打印日志。

MDC(Mapped Diagnostic Context,映射调试上下文)是log4j和logback提供的一种方便在多线程条件下记录日志的功能。

实现思路是在一个请求开始时,将请求相关的上下文信息(例如客户ID、客户的IP地址、sessionId、请求参数等)添加到MDC,然后配置好logback-spring.xml,则Logback组件将会在每条日志中打印出存放到MDC的信息,从而实现一个ID贯穿用户的所有操作。

二、代码实战

新建一个spring boot项目spring-boot-log,按照下面步骤操作。

  1. 新建日志拦截器

日志拦截器在请求开始获取用户的sessionId,当然也可以生成一个UUID,生成后存放到MDC中。

SessionInterceptor代码如下:

  1. /**
  2. * 日志拦截器
  3. * @Author: Java碎碎念
  4. *
  5. */
  6. public class SessionInterceptor extends HandlerInterceptorAdapter {
  7. /**
  8. * 会话ID
  9. */
  10. private final static String SESSION_KEY = "sessionId";
  11. @Override
  12. public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1,
  13. Object arg2, ModelAndView arg3) throws Exception {
  14. }
  15. @Override
  16. public boolean preHandle(HttpServletRequest request,
  17. HttpServletResponse response, Object handler) throws Exception {
  18. // String token = UUID.randomUUID().toString().replaceAll("-","");
  19. //本例测试使用sessionId,也可以使用UUID等
  20. String token = request.getSession().getId();
  21. MDC.put(SESSION_KEY, token);
  22. return true;
  23. }
  24. @Override
  25. public void afterCompletion(HttpServletRequest arg0,
  26. HttpServletResponse arg1, Object arg2, Exception arg3)
  27. throws Exception {
  28. // 删除
  29. MDC.remove(SESSION_KEY);
  30. }
  31. }
  1. 新建配置类

新建InterceptorConfig,注册刚才的日志拦截器。

InterceptorConfig代码如下:

  1. @Configuration
  2. public class InterceptorConfig implements WebMvcConfigurer {
  3. @Bean
  4. public SessionInterceptor getSessionInterceptor() {
  5. return new SessionInterceptor();
  6. }
  7. @Override
  8. public void addInterceptors(InterceptorRegistry registry) {
  9. registry.addInterceptor(getSessionInterceptor()).addPathPatterns("/*");
  10. }
  11. }
  1. 修改logback-spring.xml

配置logback-spring.xml,获取日志拦截器添加的sessionId并打印到日志中,配置文件中获取方式如下:

  1. %X{sessionId}

本例中打印sessionId到控制台和文件,完整配置如下:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <configuration>
  3. <property name="log.base" value="./log/logback"/>
  4. <appender name="stdout" class="ch.qos.logback.core.ConsoleAppender">
  5. <encoder>
  6. <pattern> %date [%thread] [%X{sessionId}] %-5level %logger{80} - %msg%n
  7. </pattern>
  8. </encoder>
  9. </appender>
  10. <appender name="logfile"
  11. class="ch.qos.logback.core.rolling.RollingFileAppender">
  12. <File>${log.base}.log</File>
  13. <rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
  14. <FileNamePattern>${log.base}.%d{yyyy -MM-dd}.log.zip</FileNamePattern>
  15. </rollingPolicy>
  16. <encoder>
  17. <pattern> %date [%thread] [%X{sessionId}] %-5level %logger{80} - %msg%n
  18. </pattern>
  19. </encoder>
  20. </appender>
  21. <logger name="com.sample" level="TRACE"/>
  22. <root>
  23. <level value="INFO"/>
  24. <appender-ref ref="stdout"/>
  25. <appender-ref ref="logfile"/>
  26. </root>
  27. </configuration>
  1. 添加controller

新建TestLogController,打印日志。

代码如下:

  1. @RestController
  2. public class TestLogController {
  3. Logger log = LoggerFactory.getLogger(getClass());
  4. /**
  5. * 测试登录
  6. */
  7. @RequestMapping(value = "/testLogin")
  8. public String testLogin() {
  9. log.info("用户登录成功!");
  10. return "ok";
  11. }
  12. /**
  13. * 测试下单
  14. */
  15. @RequestMapping(value = "/testNewOrder")
  16. public String testNewOrder() {
  17. log.info("用户创建了订单!");
  18. log.info("请求完成,返回ok!");
  19. return "ok";
  20. }
  21. /**
  22. * 测试购买
  23. */
  24. @RequestMapping(value = "/testPay")
  25. public String testPay() {
  26. log.info("用户付款!");
  27. return "ok";
  28. }
  29. }

三、测试

打开浏览器连续访问接口testLogin、testNewOrder和testPay,模拟用户登录、下单、付款操作,控制台和文件中打印的日志中已经包含了sessonId信息,打印的结果如下:

  1. [http-nio-8888-exec-1] [CB8E7DB250A31F2BE6C05B30633B9A95] INFO com.example.springbootlog.controller.TestLogController - 用户登录成功!
  2. [http-nio-8888-exec-2] [CB8E7DB250A31F2BE6C05B30633B9A95] INFO com.example.springbootlog.controller.TestLogController - 用户创建了订单!
  3. [http-nio-8888-exec-2] [CB8E7DB250A31F2BE6C05B30633B9A95] INFO com.example.springbootlog.controller.TestLogController - 请求完成,返回ok
  4. [http-nio-8888-exec-3] [CB8E7DB250A31F2BE6C05B30633B9A95] INFO com.example.springbootlog.controller.TestLogController - 用户付款!

到此SpringBoot+Logback手写一个简单的链路追踪功能已经全部实现,有问题欢迎留言沟通哦!

完整源码地址: https://github.com/suisui2019/springboot-study

推荐阅读

1.SpringBoot中如何优雅的读取yml配置文件?

2.SpringBoot中如何灵活的实现接口数据的加解密功能?

3.SpringBoot中神奇的@Enable*注解?

4.Java中Integer.parseInt和Integer.valueOf,你还傻傻分不清吗?

5.SpringCloud系列-整合Hystrix的两种方式


限时领取免费Java相关资料,涵盖了Java、Redis、MongoDB、MySQL、Zookeeper、Spring Cloud、Dubbo/Kafka、Hadoop、Hbase、Flink等高并发分布式、大数据、机器学习等技术。

关注下方公众号即可免费领取:

利用SpringBoot+Logback手写一个简单的链路追踪的更多相关文章

  1. 手写一个简单的starter组件

    spring-boot中有很多第三方包,都封装成starter组件,在maven中引用后,启动springBoot项目时会自动装配到spring ioc容器中. 思考: 为什么我们springBoot ...

  2. 手写一个简单的ElasticSearch SQL转换器(一)

    一.前言 之前有个需求,是使ElasticSearch支持使用SQL进行简单查询,较新版本的ES已经支持该特性(不过貌似还是实验性质的?) ,而且git上也有elasticsearch-sql 插件, ...

  3. 手写一个简单到SpirngMVC框架

    spring对于java程序员来说,无疑就是吃饭到筷子.在每次编程工作到时候,我们几乎都离不开它,相信无论过去,还是现在或是未来到一段时间,它仍会扮演着重要到角色.自己对spring有一定的自我见解, ...

  4. 手写一个简单的HashMap

    HashMap简介 HashMap是Java中一中非常常用的数据结构,也基本是面试中的"必考题".它实现了基于"K-V"形式的键值对的高效存取.JDK1.7之前 ...

  5. 手写一个简单版的SpringMVC

    一 写在前面 这是自己实现一个简单的具有SpringMVC功能的小Demo,主要实现效果是; 自己定义的实现效果是通过浏览器地址传一个name参数,打印“my name is”+name参数.不使用S ...

  6. 如何利用C# Roslyn编译器写一个简单的代码提示/错误检查?

    OK, 废话不多说,这些天在写C#代码时突然对于IDE提示有了一些想法,之前也有了解过,不过并没有深入. 先看个截图: 一段再简单不过的代码了,大家注意看到 count 字段下面的绿色波浪线了吗,我们 ...

  7. (二)如何利用C# Roslyn编译器写一个简单的代码提示/错误检查?

    上一篇我们讲了如何建立一个简单的Roslyn分析项目如分析检查我们的代码. 今天我们主要介绍各个项目中具体的作用以及可视化分析工具. 还是这种截图,可以看到解决方案下一共有三个项目. Analyzer ...

  8. jquery 手写一个简单浮窗的反面教材

    前言 初学jquery写的代码,陈年往事回忆一下. 正文 介绍一下大体思路 思路: 1.需要控制一块区域,这块区域一开始是隐藏的. 2.这个区域需要关闭按钮,同时我需要写绑定事件,关闭的时候让这块区域 ...

  9. socket手写一个简单的web服务端

    直接进入正题吧,下面的代码都是我在pycharm中写好,再粘贴上来的 import socket server = socket.socket() server.bind(('127.0.0.1', ...

随机推荐

  1. tomcat下的路径问题

    在tomcat下 如果是根据类装载器获得某个需要修改的文件路径 就有可能在web项目部署的时候存在问题 比如这里有一个测试 package Junit.test; public class test ...

  2. Wordpress SEO

    Wordpress SEO 安装插件 Baidu Sitemap Generator, 作者 柳城, 主要用于按照配置参数生成 sitemap.xml 网站地图. 设置路径 设置 => Baid ...

  3. 初识PE文件结构

    前言 目前网络上有关PE文件结构说明的文章太多了,自己的这篇文章只是单纯的记录自己对PE文件结构的学习.理解和总结. 基础概念 PE(Portable Executable:可移植的执行体)是Win3 ...

  4. unzip 命令指定解压路径

    在使用unzip进行文件包解压,可以用来解压zip/jar/war包类型,有时解压时需要解压到指定路径时可以使用参数 -d 来指定,例如: unzip services-bak.jar -d ./we ...

  5. javascript中字符串对象常用的方法和属性

    前言 字符串是一种非常重要的数据类型,在Java等面向对象编程语言中,它代表对象类型,而在javascript中它却是一种基本数据类型,在开发的领域中,我们经常会碰到,无论是前端还是后台.比如后台验证 ...

  6. vue2.0项目记住密码和用户名实例

    的今天突来兴致,试了一下将用户名和密码存在cookie和localStorage里如何实现:从代码难易程度来讲,果断选择了将用户名和密码存在localStorage里面.当然菜鸟上这么说的,楼下. 也 ...

  7. 同时支持EF+Dapper的混合仓储,助你快速搭建数据访问层

    背景 17年开始,公司开始向DotNet Core转型,面对ORM工具的选型,当时围绕Dapper和EF发生了激烈的讨论.项目团队更加关注快速交付,他们主张使用EF这种能快速开发的ORM工具:而在线业 ...

  8. 使用 chart 部署 skywalking

    使用 chart 部署 skywalking 本文主要讲述的是如何使用 Helm Charts 将 SkyWalking 部署到 Kubernetes 集群中,相关文档可以参考skywalking-k ...

  9. 移动端适配 rem 设置

    refresh();    window.onresize = function(){      setTimeout(function(){        refresh();      },10) ...

  10. Maven私服Nexus的搭建

    # Maven私服Nexus的搭建 ## 私服存在的合理性 Maven中的依赖是从服务器仓库中下载的,Maven的仓库只有两大类: - 1) 本地仓库 - 2) 远程仓库,其中在远程仓库中又分成了3种 ...