MDC简介

MDC简介一 MDC 介绍 MDC MappedDiagno 映射调试上下文 是 log4j 和 logback 提供的一种方便在多线程条件下记录日志的功能 某些应用程序采用多线程的方式来处理多个用户的请求 在一个用户的使用过程中 可能有多个不同的线程来进行处理 典型的例子是 Web 应用服务器 当用户访问某个页面时 应用服务器可能会创建一个新的线程来处理该请求 也可能从线

一:MDC介绍

  MDC(Mapped Diagnostic Context,映射调试上下文)是 log4j 和 logback 提供的一种方便在多线程条件下记录日志的功能。某些应用程序采用多线程的方式来处理多个用户的请求。在一个用户的使用过程中,可能有多个不同的线程来进行处理。典型的例子是 Web 应用服务器。当用户访问某个页面时,应用服务器可能会创建一个新的线程来处理该请求,也可能从线程池中复用已有的线程。在一个用户的会话存续期间,可能有多个线程处理过该用户的请求。这使得比较难以区分不同用户所对应的日志。当需要追踪某个用户在系统中的相关日志记录时,就会变得很麻烦。

  一种解决的办法是采用自定义的日志格式,把用户的信息采用某种方式编码在日志记录中。这种方式的问题在于要求在每个使用日志记录器的类中,都可以访问到用户相关的信息。这样才可能在记录日志时使用。这样的条件通常是比较难以满足的。MDC 的作用是解决这个问题。

  MDC 可以看成是一个与当前线程绑定的哈希表,可以往其中添加键值对。MDC 中包含的内容可以被同一线程中执行的代码所访问。当前线程的子线程会继承其父线程中的 MDC 的内容。当需要记录日志时,只需要从 MDC 中获取所需的信息即可。MDC 的内容则由程序在适当的时候保存进去。对于一个 Web 应用来说,通常是在请求被处理的最开始保存这些数据。

  为了验证MDC的正确性,我写了个简单的多线程程序,代码如下:

查看文本打印

  1. import org.apache.log4j.MDC;  
  2.   
  3. public class ThreadTest extends Thread {  
  4.     private int i ;  
  5.       
  6.     public ThreadTest(){  
  7.     }  
  8.       
  9.     public ThreadTest(int i){  
  10.         this.i = i;  
  11.     }  
  12.       
  13.     public void run(){  
  14.         System.out.println(++i);  
  15.         MDC.put(“username”, i);  
  16.       
  17.         for (int j = 0; j < 100; j++) {  
  18.             System.out.println(“aaa” + i);  
  19.             if(j==10){  
  20.                 try {  
  21.                     this.sleep(1000);  
  22.                 } catch (InterruptedException e) {  
  23.                     e.printStackTrace();  
  24.                 }  
  25.             }  
  26.         }  
  27.         System.out.println(“run: “ + i + ”     “  + MDC.get(“username”));  
  28.     }  
  29.       
  30.     public static void main(String args[]) throws InterruptedException{  
  31.         ThreadTest t1 = new ThreadTest(1);  
  32.         t1.start();  
  33.         ThreadTest t2 = new ThreadTest(2);  
  34.         t2.start();  
  35.     }  
  36. }  

运行结果如下:

查看文本打印

  1. 2  
  2. 3  
  3. aaa3  
  4. aaa3  
  5. aaa2  
  6. aaa3  
  7. aaa2  
  8. aaa3  
  9. aaa2  
  10. aaa3  
  11. aaa2  
  12. aaa3  
  13. aaa2  
  14. aaa3  
  15. aaa2  
  16. aaa2  
  17. aaa2  
  18. aaa2  
  19. aaa2  
  20. run: 2     2  
  21. aaa3  
  22. aaa3  
  23. aaa3  
  24. run: 3     3  

  MDC类put方法:

查看文本打印

  1. public static void put(String key, Object o)  
  2.   {  
  3.     mdc.put0(key, o);  
  4.   }  
  5.   
  6.   private void put0(String key, Object o)  
  7.   {  
  8.     if (this.java1) {  
  9.       return;  
  10.     }  
  11.     Hashtable ht = (Hashtable)((ThreadLocalMap)this.tlm).get();  
  12.     if (ht == null) {  
  13.       ht = new Hashtable(7);  
  14.       ((ThreadLocalMap)this.tlm).set(ht);  
  15.     }  
  16.     ht.put(key, o);  
  17.   }  

结合类java.lang.ThreadLocal

及Thread类可以知道,MDC中的put方法其实就是讲键值对放入一个Hashtable对象中,然后赋值给当前线程的ThreadLocal.ThreadLocalMap对象,即threadLocals,这保证了各个线程的在MDC键值对的独立性。

下边为java.lang.ThreadLocal

的部分代码:

查看文本打印

  1. public class ThreadLocal

     {  
  2.       
  3.     public void set(T value) {  
  4.         Thread t = Thread.currentThread();  
  5.         ThreadLocalMap map = getMap(t);  
  6.         if (map != null)  
  7.             map.set(this, value);  
  8.         else  
  9.             createMap(t, value);  
  10.     }  
  11.   
  12.   
  13.     public T get() {  
  14.         Thread t = Thread.currentThread();  
  15.         ThreadLocalMap map = getMap(t);  
  16.         if (map != null) {  
  17.             ThreadLocalMap.Entry e = map.getEntry(this);  
  18.             if (e != null)  
  19.                 return (T)e.value;  
  20.         }  
  21.         return setInitialValue();  
  22.     }  
  23.   
  24.     ThreadLocalMap getMap(Thread t) {  
  25.         return t.threadLocals;  
  26.     }  
  27.       
  28. }  

Thread类的部分代码:

查看文本 打印

  1. public class Thread implements Runnable {  
  2.     ThreadLocal.ThreadLocalMap threadLocals = null;  
  3.   
  4.     ………………….  
  5.     …………………….  
  6.   
  7.     static class ThreadLocalMap { //ThreadLocalMap为Thread类的内部类  
  8.   
  9.     }  
  10. }  

 

二:日志聚合与分析(摘自:http://www.ibm.com/developerworks/cn/java/j-lo-practicelog/index.html) — 此部分是为了方便自己查阅,从参考资料中摘出来放到这里的

在程序中正确的地方输出合适的日志消息,只是合理使用日志的第一步。日志记录的真正作用在于当有问题发生时,能够帮助开发人员很快的定位问题所在。不过一个实用的系统通常由很多个不同的部分组成。这其中包括所开发的程序本身,也包括所依赖的第三方应用程序。以一个典型的电子商务网站为例,除了程序本身,还包括所依赖的底层操作系统、应用服务器、数据库、HTTP 服务器和代理服务器和缓存等。当一个问题发生时,真正的原因可能来自程序本身,也可能来自所依赖的第三方程序。这就意味着开发人员可能需要检查不同服务器上不同应用程序的日志来确定真正的原因。

日志聚合的作用就在于可以把来自不同服务器上不同应用程序产生的日志聚合起来,存放在单一的服务器上,方便进行搜索和分析。在日志聚合方面,已经有不少成熟的开源软件可以很好的满足需求。本文中要介绍的是 logstash,一个流行的事件和日志管理开源软件。logstash 采用了一种简单的处理模式:输入 -> 过滤器 -> 输出。logstash 可以作为代理程序安装到每台需要收集日志的机器上。logstash 提供了非常多的插件来处理不同类型的数据输入。典型的包括控制台、文件和 syslog 等;对于输入的数据,可以使用过滤器来进行处理。典型的处理方式是把日志消息转换成结构化的字段;过滤之后的结果可以被输出到不同的目的地,比如 ElasticSearch、文件、电子邮件和数据库等。

Logstash 在使用起来很简单。从官方网站下载 jar 包并运行即可。在运行时需要指定一个配置文件。配置文件中定义了输入、过滤器和输出的相关配置。清单 9 给出了一个简单的 logstash 配置文件的示例。

清单 9. logstash 配置文件示例
查看文本 打印

  1. input {   
  2.  file {   
  3.    path => [ “/var/log/*.log”“/var/log/messages”“/var/log/syslog” ]   
  4.    type => ‘syslog’  
  5.  }   
  6. }   
  7.   
  8. output {   
  9.  stdout {   
  10. debug => true   
  11. debug_format => “json”   
  12.  }   
  13. }  

清单 9 中定义了 logstash 收集日志时的输入(input)和输出(output)的相关配置。输入类型是文件(file)。每种类型输入都有相应的配置。对于文件来说,需要配置的是文件的路径。对每种类型的输入,都需要指定一个类型(type)。该类型用来区分来自不同输入的记录。代码中使用的输出是控制台。配置文件完成之后,通过“java -jar logstash-1.1.13-flatjar.jar agent -f logstash-simple.conf”就可以启动 logstash。

在日志分析中,比较重要的是结构化的信息。而日志信息通常只是一段文本,其中的不同字段表示不同的含义。不同的应用程序产生的日志的格式并不相同。在分析时需要关注的是其中包含的不同字段。比如 Apache 服务器会产生与用户访问请求相关的日志。在日志中包含了访问者的各种信息,包括 IP 地址、时间、HTTP 状态码、响应内容的长度和 User Agent 字符串等信息。在 logstash 收集到日志信息之后,可以根据一定的规则把日志信息中包含的数据提取出来并命名。logstash 提供了 grok 插件可以完成这样的功能。grok 基于正则表达式来工作,同时提供了非常多的常用类型数据的提取模式,如清单 10 所示。

清单 10. 使用 grok 提取日志记录中的内容

点击查看代码清单

在经过上面 grok 插件的提取之后,Apache 访问日志被转换成包含字段 client、method、request、status、bytes 和 useragent 的格式化数据。可以根据这些字段来进行搜索。这对于分析问题和进行统计都是很有帮助的。

当日志记录通过 logstash 进行收集和处理之后,通常会把这些日志记录保存到数据库中进行分析和处理。目前比较流行的方式是保存到 ElasticSearch 中,从而可以利用 ElasticSearch 提供的索引和搜索能力来分析日志。已经有不少的开源软件在 ElasticSearch 基础之上开发出相应的日志管理功能,可以很方便的进行搜索和分析。本文中介绍的是 Graylog2。

Graylog2 由服务器和 Web 界面两部分组成。服务器负责接收日志记录并保存到 ElasticSearch 之中。Web 界面则可以查看和搜索日志,并提供其他的辅助功能。logstash 提供了插件 gelf,可以把 logstash 收集和处理过的日志记录发送到 Graylog2 的服务器。这样就可以利用 Graylog2 的 Web 界面来进行查询和分析。只需要把清单 9 中的 logstash 的配置文件中的 output 部分改成清单 11 中所示即可。

清单 11. 配置 logstash 输出到 Graylog2
查看文本 打印

  1. output {   
  2.  gelf {   
  3.    host => ‘127.0.0.1’  
  4.  }   
  5. }  

在安装 Graylog2 时需要注意,一定要安装与 Graylog2 的版本相对应的版本的 ElasticSearch,否则会出现日志记录无法保存到 ElasticSearch 的问题。本文中使用的是 Graylog2 服务器 0.11.0 版本和 ElasticSearch 0.20.4 版本。

除了 Graylog2 之外,另外一个开源软件 Kibana 也比较流行。Kibana 可以看成是 logstash 和 ElasticSearch 的 Web 界面。Kibana 提供了更加丰富的功能来显示和分析日志记录。与代码清单中的 logstash 的配置相似,只需要把输出改为 elasticsearch 就可以了。Kibana 可以自动读取 ElasticSearch 中包含的日志记录并显示。

转载自: https://blog.csdn.net/sunzhenhua0608/article/details/

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/224193.html原文链接:https://javaforall.net

(0)
上一篇 2026年3月17日 下午12:29
下一篇 2026年3月17日 下午12:30


相关推荐

  • vhr项目

    vhr项目vhr 前端编译代码 https github com lenve vhr 当前使用 node 版本 v14 7 0 npm 版本 7 21 1 在根目录执行 npminstallun perm 启动 npmrunserveh mp weixin com s biz MzI1NDY0MTkz amp mid amp idx 1 amp sn 0b0f2d22dd2d amp scene 21 we

    2026年3月18日
    3
  • 【渗透测试】密码暴力破解工具——九头蛇(hydra)使用详解及实战

    【渗透测试】密码暴力破解工具——九头蛇(hydra)使用详解及实战【渗透测试】密码暴力破解工具——九头蛇(hydra)使用详解及实战

    2022年5月2日
    58
  • 基于java的小区物业管理系统_java微服务架构

    基于java的小区物业管理系统_java微服务架构毕设项目——智慧小区系统项目初衷(最真实版)系统技术分析前端界面后端及数据库系统功能介绍小区业主端物业人员端系统界面展示登录界面首页信息列表界面新增界面删除提示界面修改界面查询界面业主查看物流信息界面小结项目初衷(最真实版)其实一开始,笔者只想做一个最最简单的管理系统,通篇只有增删改查的那种,但是马上就被老师批斗说工作量太少了,不得已最后做了个前台后台的完整版。不仅有后台的物业管理,也有前台的对小区业主服务,只不过都是简易版,本科毕设,大家宽容哈。系统技术分析前端界面后端及数据库系统功能介绍

    2022年10月18日
    5
  • php 下载文件的头信息 Determine Content Type

    php 下载文件的头信息 Determine Content Type

    2021年10月14日
    57
  • Nginx sendfile原理详解[通俗易懂]

    Nginx sendfile原理详解[通俗易懂]配置语法语法:sendfileon|off;默认值:sendfileoff;上下文:http,server,location,ifinlocation说明sendfile值为on,指定使用sendfile系统调用来传输文件。sendfile系统调用在两个文件描述符之间直接传递数据(完全在内核中操作),从而避免了数据在内核缓冲区和用户缓冲区之间的拷贝,操作效率很高,被…

    2022年5月16日
    205
  • Java XML解析工具类

    Java XML解析工具类JavaXML解析工具类Java解析XML的方式有很多,这里不一一说明了,利用三方jar包,实现了一个XML工具类本身是有个需求,讲三方公司的XML请求文件中的Response里的属性赋值成java对象,没做完不需要了,只做到了解析,后续本来想利用解析出来的key进行遍历,利用FTL模板生成Java文件,有这个需求的小伙伴可以延续这个工具类。1.看看XML<?xmlversion=”1.0″encoding=”utf-8″?><!–每个协议有Request或Resp

    2022年7月21日
    16

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号