`
wsmajunfeng
  • 浏览: 496850 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

log4j MDC NDC应用场景

 
阅读更多

NDC Nested Diagnostic Context )和 MDC Mapped Diagnostic Context )是 log4j 种非常有用的两个类,它们用于存储应用程序的上下文信息( context infomation ),从而便于在 log 中使用这些上下文信息。

 

NDC的实现是用hashtable来存储每个线程的stack信息,这个stack是每个线程可以设置当前线程的request的相关信息,然后当前线 程在处理过程中只要在log4j配置打印出%x的信息,那么当前线程的整个stack信息就会在log4j打印日志的时候也会都打印出来,这样可以很好的 跟踪当前request的用户行为功能。

MDC的实现是使用threadlocal来保存每个线程的Hashtable的类似map的信息,其他功能类似。

 

NDC的实现代码:

 

[java] view plain copy
  1. public   class  NDC {  
  2.   
  3.   
  4.   static  Hashtable ht =  new  Hashtable();  
  5.   
  6.     
  7.   private   static  Stack getCurrentStack() {  
  8.       if  (ht !=  null ) {  
  9.           return  (Stack) ht.get(Thread.currentThread());  
  10.       }  
  11.       return   null ;  
  12.   }  
  13.   
  14.   public   
  15.   static   
  16.   String pop() {  
  17.     Stack stack = getCurrentStack();  
  18.     if (stack !=  null  && !stack.isEmpty())   
  19.       return  ((DiagnosticContext) stack.pop()).message;  
  20.     else   
  21.       return   "" ;  
  22.   }  
  23.   public   
  24.   static   
  25.   String peek() {  
  26.     Stack stack = getCurrentStack();  
  27.     if (stack !=  null  && !stack.isEmpty())  
  28.       return  ((DiagnosticContext) stack.peek()).message;  
  29.     else   
  30.       return   "" ;  
  31.   }  
  32.   public   
  33.   static   
  34.   void  push(String message) {  
  35.     Stack stack = getCurrentStack();  
  36.         
  37.     if (stack ==  null ) {  
  38.       DiagnosticContext dc = new  DiagnosticContext(message,  null );        
  39.       stack = new  Stack();  
  40.       Thread key = Thread.currentThread();  
  41.       ht.put(key, stack);  
  42.       stack.push(dc);  
  43.     } else   if  (stack.isEmpty()) {  
  44.       DiagnosticContext dc = new  DiagnosticContext(message,  null );              
  45.       stack.push(dc);  
  46.     } else  {  
  47.       DiagnosticContext parent = (DiagnosticContext) stack.peek();  
  48.       stack.push(new  DiagnosticContext(message, parent));  
  49.     }      
  50.   }  

 

MDC的实现:

 

[java] view plain copy
  1. public   class  MDC {  
  2.     
  3.   final   static  MDC mdc =  new  MDC();  
  4.     
  5.   static   final   int  HT_SIZE =  7 ;  
  6.   
  7.   boolean  java1;  
  8.     
  9.   Object tlm;  
  10.   
  11.   private  Method removeMethod;  
  12.   
  13.   private   
  14.   MDC() {  
  15.     java1 = Loader.isJava1();  
  16.     if (!java1) {  
  17.       tlm = new  ThreadLocalMap();  
  18.     }  
  19.   
  20.     try  {  
  21.       removeMethod = ThreadLocal.class .getMethod( "remove" null );  
  22.     } catch  (NoSuchMethodException e) {  
  23.       // don't do anything - java prior 1.5   
  24.     }  
  25.   }  
  26.   
  27.    
  28.    */  
  29.   static   
  30.   public   
  31.   void  put(String key, Object o) {  
  32.      if  (mdc !=  null ) {  
  33.          mdc.put0(key, o);  
  34.      }  
  35.   }  
  36.   
  37.   static    
  38.   public   
  39.   Object get(String key) {  
  40.     if  (mdc !=  null ) {  
  41.         return  mdc.get0(key);  
  42.     }  
  43.     return   null ;  
  44.   }  
  45.   
  46.   static    
  47.   public   
  48.   void  remove(String key) {  
  49.     if  (mdc !=  null ) {  
  50.         mdc.remove0(key);  
  51.     }  
  52.   }  
  53.   
  54.   
  55.   public   static  Hashtable getContext() {  
  56.     if  (mdc !=  null ) {  
  57.         return  mdc.getContext0();  
  58.     } else  {  
  59.         return   null ;  
  60.     }  
  61.   }  
  62.   
  63.   
  64.   public   static   void  clear() {  
  65.     if  (mdc !=  null ) {  
  66.         mdc.clear0();  
  67.     }  
  68.   }  
  69.   
  70.   
  71.   private   
  72.   void  put0(String key, Object o) {  
  73.     if (java1 || tlm ==  null ) {  
  74.       return ;  
  75.     } else  {  
  76.       Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();  
  77.       if (ht ==  null ) {  
  78.         ht = new  Hashtable(HT_SIZE);  
  79.         ((ThreadLocalMap)tlm).set(ht);  
  80.       }      
  81.       ht.put(key, o);  
  82.     }  
  83.   }  
  84.     
  85.   private   
  86.   Object get0(String key) {  
  87.     if (java1 || tlm ==  null ) {  
  88.       return   null ;  
  89.     } else  {         
  90.       Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();  
  91.       if (ht !=  null  && key !=  null ) {  
  92.         return  ht.get(key);  
  93.       } else  {  
  94.         return   null ;  
  95.       }  
  96.     }  
  97.   }  
  98.   
  99.   private   
  100.   void  remove0(String key) {  
  101.     if (!java1 && tlm !=  null ) {  
  102.       Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();  
  103.       if (ht !=  null ) {  
  104.         ht.remove(key);  
  105.         // clean up if this was the last key   
  106.         if  (ht.isEmpty()) {  
  107.           clear0();  
  108.         }  
  109.       }   
  110.     }  
  111.   }  
  112.   
  113.   
  114.   private   
  115.   Hashtable getContext0() {  
  116.      if (java1 || tlm ==  null ) {  
  117.       return   null ;  
  118.     } else  {         
  119.       return  (Hashtable) ((ThreadLocalMap)tlm).get();  
  120.     }  
  121.   }  
  122.   
  123.   private   
  124.   void  clear0() {  
  125.     if (!java1 && tlm !=  null ) {  
  126.       Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();  
  127.       if (ht !=  null ) {  
  128.         ht.clear();  
  129.       }  
  130.       if (removeMethod !=  null ) {  
  131.           // java 1.3/1.4 does not have remove - will suffer from a memory leak   
  132.           try  {  
  133.             removeMethod.invoke(tlm, null );  
  134.           } catch  (IllegalAccessException e) {  
  135.             // should not happen   
  136.           } catch  (InvocationTargetException e) {  
  137.             // should not happen   
  138.           }  
  139.       }  
  140.     }  
  141.   }  
  142.   
  143. }  

 

在webx框架中对于log4j的MDC的处理:

先配置一个filter,这个filter是放置web.xml的最前面

[java] view plain copy
  1. protected   void  populateMDC(Map<String, String> mdc) {  
  2.     // GET or POST   
  3.     putMDC(mdc, MDC_METHOD, request.getMethod());  
  4.   
  5.       
  6.     StringBuffer requestURL = request.getRequestURL();  
  7.     String queryString = trimToNull(request.getQueryString());  
  8.   
  9.     putMDC(mdc, MDC_REQUEST_URL, getRequestURL(requestURL, null ));  
  10.     putMDC(mdc, MDC_REQUEST_URL_WITH_QUERY_STRING, getRequestURL(requestURL, queryString));  
  11.   
  12.       
  13.     String requestURI = request.getRequestURI();  
  14.     String requestURIWithQueryString = queryString == null  ? requestURI : requestURI +  "?"  + queryString;  
  15.   
  16.     putMDC(mdc, MDC_REQUEST_URI, requestURI);  
  17.     putMDC(mdc, MDC_REQUEST_URI_WITH_QUERY_STRING, requestURIWithQueryString);  
  18.     putMDC(mdc, MDC_QUERY_STRING, queryString);  
  19.   
  20.     // client info   
  21.     putMDC(mdc, MDC_REMOTE_HOST, request.getRemoteHost());  
  22.     putMDC(mdc, MDC_REMOTE_ADDR, request.getRemoteAddr());  
  23.   
  24.     // user agent   
  25.     putMDC(mdc, MDC_USER_AGENT, request.getHeader("User-Agent" ));  
  26.   
  27.     // referrer   
  28.     putMDC(mdc, MDC_REFERRER, request.getHeader("Referer" ));  
  29.   
  30.     // cookies   
  31.     Cookie[] cookies = request.getCookies();  
  32.     List<String> names = emptyList();  
  33.   
  34.     if  (cookies !=  null ) {  
  35.         names = createArrayList(cookies.length);  
  36.   
  37.         for  (Cookie cookie : cookies) {  
  38.             names.add(cookie.getName());  
  39.             putMDC(mdc, MDC_COOKIE_PREFIX + cookie.getName(), cookie.getValue());  
  40.         }  
  41.   
  42.         sort(names);  
  43.     }  
  44.   
  45.     putMDC(mdc, MDC_COOKIES, names.toString());  
  46. }  


在finally中记住cleanMDC,否则可能会造成OOM。

 

[java] view plain copy
  1.    try  {   helper.setLoggingContext();  
  2.   
  3.             chain.doFilter(request, response);  
  4.         } finally  {  
  5.             helper.clearLoggingContext();  
  6.         }  

 

 

在 NDC 简介部分,我们曾经说过,%x 表示会在每个日志行上打印当前 NDC 上下文。

MDC  %X{remoteAddr}  {remoteAddr} 表示对应map中的remoteAddr的值

配置log4j:

[html] view plain copy
  1. < param   name = "ConversionPattern"   value = "%d{yyyy-MM-dd HH\:mm\:ss} %X{remoteAddr} %X{requestURI} %X{referrer} %X{userAgent} %c %c - %m%n " />   


打 印日志如下:2012-05-22 22:16:30 127.0.0.1 /cta/index.htm Mozilla/5.0 (Windows NT 6.1) AppleWebKit/535.19 (KHTML, like Gecko) Chrome/18.0.1025.168 Safari/535.19 com.alibaba.citrus.webx.impl.WebxRootControllerImpl com.alibaba.citrus.webx.impl.WebxRootControllerImpl - Error occurred while process request /cta/index.htm

 

[html] view plain copy
  1. java.lang.NullPointerException  
  2.     at com.alibaba.citrus.webx.impl.WebxControllerImpl.service(WebxControllerImpl.java:42)  
  3.     at com.alibaba.citrus.webx.impl.WebxRootControllerImpl.handleRequest(WebxRootControllerImpl.java:53)  
  4.     at com.alibaba.citrus.webx.support.AbstractWebxRootController.service(AbstractWebxRootController.java:156)  
  5.     at com.alibaba.citrus.webx.servlet.WebxFrameworkFilter.doFilter(WebxFrameworkFilter.java:141)  
  6.     at com.alibaba.citrus.webx.servlet.FilterBean.doFilter(FilterBean.java:164)  
  7.     at org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1322)  
  8.     at com.alibaba.citrus.webx.servlet.SetLoggingContextFilter.doFilter(SetLoggingContextFilter.java:62)  
  9.     at com.alibaba.citrus.webx.servlet.FilterBean.doFilter(FilterBean.java:164)  
  10.     at org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1322) 
分享到:
评论

相关推荐

    log4j简单使用

    Log4j是Apache软件基金会开发的一个开源项目,它为Java应用程序提供了一个灵活的日志系统,允许开发者自定义日志级别、输出格式以及存储位置等,以满足不同场景的需求。 描述中的"NULL"意味着没有额外的具体信息,...

    Log4J完整说明和配置

    以上配置覆盖了Log4J常用的几种Appender类型及其配置参数,能够满足大多数应用场景的需求。开发者可以根据实际需求调整各个Appender的配置,以实现更加精细化的日志管理。此外,Log4J还提供了丰富的扩展机制,如MDC...

    apache-log4j-2.4.1-bin.zip

    总的来说,Apache Log4j 2是一个强大且灵活的日志框架,它不仅提供了高效的日志记录能力,还通过丰富的配置选项和插件支持,满足了各种复杂应用场景的需求。在实际项目中,你可以根据“apache-log4j-2.4.1-bin.zip”...

    log4j-slf4j-impl.zip

    5. **MDC与NDC**:Log4j-SLF4J-Impl支持Mapped Diagnostic Context (MDC) 和Nested Diagnostic Context (NDC),为日志信息添加上下文信息,有助于追踪和诊断问题。 四、集成与使用 集成Log4j-SLF4J-Impl通常涉及...

    log4j-2.11.1.jar

    Apache Log4j 是一个广泛使用的开源日志记录框架,为Java应用程序提供了强大的日志处理能力。在2.11.1这个版本中,Log4j 继续提供高效、灵活的日志记录解决方案,旨在满足各种开发需求。本文将深入探讨Log4j 2.11.1...

    log4j使用实例.zip

    对于高并发场景,Log4j还提供了异步日志功能,通过使用AsyncAppender,可以避免日志输出对主线程的影响,提高应用性能。 7. **MDC与NDC** 异常跟踪时,可以使用Mapped Diagnostic Context (MDC) 和 Nested ...

    log4j入门、详解

    总结来说,Log4j是一个强大且灵活的日志框架,它的广泛使用证明了其在Java应用日志管理中的价值。通过深入理解和正确配置,Log4j可以帮助开发者更好地监控程序状态,定位和解决问题,从而提升软件的稳定性和可靠性。...

    log4j记录Debug日志

    Log4j是Apache组织开发的一款广泛使用的Java日志框架,主要功能是用于记录应用程序运行过程中的各种日志信息。在Java编程中,日志记录是非常重要的一环,它可以帮助开发者追踪程序运行状态,定位错误,优化性能,...

    Log4j2使用案例

    **Log4j2使用案例** 在Java开发中,日志记录是不可或缺的一部分,它帮助开发者追踪程序运行状态、定位问题和进行性能分析。Log4j2作为Apache的一个顶级项目,是Log4j的升级版,提供了更高效、更灵活的日志处理能力...

    西安野马计算机培训学校LOG4J讲义

    LOG4J还支持MDC(Mapped Diagnostic Context)和NDC(Nested Diagnostic Context),它们提供了上下文信息,帮助追踪特定请求或线程的日志。此外,LOG4J的异步日志记录功能可以提高日志处理效率,尤其在高并发场景下...

    log4j详解.txt

    - **NDC与MDC**:NDC和MDC是Log4j提供的两个重要的上下文管理机制。NDC用于在多层调用中保持调用链的上下文信息,MDC则用于存储线程特有的键值对数据,便于日志分析。 - **动态配置**:Log4j支持在运行时动态修改...

    log4j-1.2.15

    除了基本的日志功能,log4j-1.2.15还支持一些高级特性,如异步日志记录(使用AsyncAppender)、日志事件的自定义处理(通过编写自定义Appender和Layout)、MDC(Mapped Diagnostic Context)和NDC(Nested ...

    log4j示例(附带对应jar包)

    本示例包含Log4j的使用说明及对应的jar包,旨在帮助开发者快速理解和应用Log4j。 **一、Log4j的基本组件** 1. **Logger**: 日志记录器,是日志系统的核心,负责生成日志信息。每个类通常都有一个与之关联的Logger...

    log4j.rar,避免在官网上找不到

    7. **日志事件上下文**:Log4j2引入了MDC(Mapped Diagnostic Context)和NDC(Nested Diagnostic Context),用于存储和传递与日志事件相关的上下文信息。 综上所述,Log4j是一个功能强大且高度可配置的日志框架,...

    日志配置到文件,数据库

    - `log4j.appender.FILE.layout.ConversionPattern=[framework]%d-%c-%-4r [%t]%-5p%c%x-%m%n`:文件输出的具体格式与控制台略有不同,例如添加了`%x`(NDC)等元素。 **4. 数据库输出配置** - `log4j.appender....

    log4j配置 ACCP教师录屏

    10. **使用场景**:Log4j适用于各种Java应用,尤其在调试、故障排查、性能监控等方面。 **源代码分析** 在提供的压缩包中,可能包含了示例代码,这些代码展示了如何在Java项目中集成和使用Log4j。通过分析这些代码...

    apache-log4j-1.2.15

    在使用Hibernate这样的ORM框架时,Log4j经常被用来记录应用的运行状态和调试信息。 **1. 日志框架的重要性** 日志是软件开发中不可或缺的一部分,它帮助开发者追踪程序运行时的问题,进行性能分析,以及优化代码。...

    log4j 1.2.12.zip

    Log4j是Apache软件基金会开发的一个开源日志组件,它为Java应用程序提供了灵活的日志记录功能。Log4j 1.2.12是其1.2系列的一个稳定版本,包含了一系列优化和改进,旨在提高性能和稳定性。 2. **核心组件** - **...

    log4j学习

    在Java编程领域,日志记录是不可或缺的一部分,而Log4j是一个广泛使用的开源日志框架,它为应用程序提供了一种灵活的日志记录方式。这个压缩包文件“log4jTest”很可能包含了与Log4j相关的测试代码或示例,帮助我们...

    Log4j使用指南.docx

    Log4j 是一款广泛使用的Java日志记录框架,它的主要目标是为应用程序提供灵活的日志输出控制,方便调试和监控。以下是对Log4j及其相关工具的详细解释: 1. **日志介绍**: 日志是记录软件运行时状态的重要手段,它...

Global site tag (gtag.js) - Google Analytics