zoukankan      html  css  js  c++  java
  • log4j MDC NDC应用场景

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

     

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

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

     

    NDC的实现代码:

    public class NDC {
    
    
      static Hashtable ht = new Hashtable();
    
      
      private static Stack getCurrentStack() {
          if (ht != null) {
              return (Stack) ht.get(Thread.currentThread());
          }
          return null;
      }
    
      public
      static
      String pop() {
        Stack stack = getCurrentStack();
        if(stack != null && !stack.isEmpty()) 
          return ((DiagnosticContext) stack.pop()).message;
        else
          return "";
      }
      public
      static
      String peek() {
        Stack stack = getCurrentStack();
        if(stack != null && !stack.isEmpty())
          return ((DiagnosticContext) stack.peek()).message;
        else
          return "";
      }
      public
      static
      void push(String message) {
        Stack stack = getCurrentStack();
          
        if(stack == null) {
          DiagnosticContext dc = new DiagnosticContext(message, null);      
          stack = new Stack();
          Thread key = Thread.currentThread();
          ht.put(key, stack);
          stack.push(dc);
        } else if (stack.isEmpty()) {
          DiagnosticContext dc = new DiagnosticContext(message, null);            
          stack.push(dc);
        } else {
          DiagnosticContext parent = (DiagnosticContext) stack.peek();
          stack.push(new DiagnosticContext(message, parent));
        }    
      }
    

    MDC的实现:

     

    public class MDC {
      
      final static MDC mdc = new MDC();
      
      static final int HT_SIZE = 7;
    
      boolean java1;
      
      Object tlm;
    
      private Method removeMethod;
    
      private
      MDC() {
        java1 = Loader.isJava1();
        if(!java1) {
          tlm = new ThreadLocalMap();
        }
    
        try {
          removeMethod = ThreadLocal.class.getMethod("remove", null);
        } catch (NoSuchMethodException e) {
          // don't do anything - java prior 1.5
        }
      }
    
     
       */
      static
      public
      void put(String key, Object o) {
         if (mdc != null) {
             mdc.put0(key, o);
         }
      }
    
      static 
      public
      Object get(String key) {
        if (mdc != null) {
            return mdc.get0(key);
        }
        return null;
      }
    
      static 
      public
      void remove(String key) {
        if (mdc != null) {
            mdc.remove0(key);
        }
      }
    
    
      public static Hashtable getContext() {
        if (mdc != null) {
            return mdc.getContext0();
        } else {
            return null;
        }
      }
    
    
      public static void clear() {
        if (mdc != null) {
            mdc.clear0();
        }
      }
    
    
      private
      void put0(String key, Object o) {
        if(java1 || tlm == null) {
          return;
        } else {
          Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();
          if(ht == null) {
            ht = new Hashtable(HT_SIZE);
            ((ThreadLocalMap)tlm).set(ht);
          }    
          ht.put(key, o);
        }
      }
      
      private
      Object get0(String key) {
        if(java1 || tlm == null) {
          return null;
        } else {       
          Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();
          if(ht != null && key != null) {
            return ht.get(key);
          } else {
            return null;
          }
        }
      }
    
      private
      void remove0(String key) {
        if(!java1 && tlm != null) {
          Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();
          if(ht != null) {
            ht.remove(key);
            // clean up if this was the last key
            if (ht.isEmpty()) {
              clear0();
            }
          } 
        }
      }
    
    
      private
      Hashtable getContext0() {
         if(java1 || tlm == null) {
          return null;
        } else {       
          return (Hashtable) ((ThreadLocalMap)tlm).get();
        }
      }
    
      private
      void clear0() {
        if(!java1 && tlm != null) {
          Hashtable ht = (Hashtable) ((ThreadLocalMap)tlm).get();
          if(ht != null) {
            ht.clear();
          }
          if(removeMethod != null) {
              // java 1.3/1.4 does not have remove - will suffer from a memory leak
              try {
                removeMethod.invoke(tlm, null);
              } catch (IllegalAccessException e) {
                // should not happen
              } catch (InvocationTargetException e) {
                // should not happen
              }
          }
        }
      }
    
    }
    

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

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

        protected void populateMDC(Map<String, String> mdc) {
            // GET or POST
            putMDC(mdc, MDC_METHOD, request.getMethod());
    
            
            StringBuffer requestURL = request.getRequestURL();
            String queryString = trimToNull(request.getQueryString());
    
            putMDC(mdc, MDC_REQUEST_URL, getRequestURL(requestURL, null));
            putMDC(mdc, MDC_REQUEST_URL_WITH_QUERY_STRING, getRequestURL(requestURL, queryString));
    
            
            String requestURI = request.getRequestURI();
            String requestURIWithQueryString = queryString == null ? requestURI : requestURI + "?" + queryString;
    
            putMDC(mdc, MDC_REQUEST_URI, requestURI);
            putMDC(mdc, MDC_REQUEST_URI_WITH_QUERY_STRING, requestURIWithQueryString);
            putMDC(mdc, MDC_QUERY_STRING, queryString);
    
            // client info
            putMDC(mdc, MDC_REMOTE_HOST, request.getRemoteHost());
            putMDC(mdc, MDC_REMOTE_ADDR, request.getRemoteAddr());
    
            // user agent
            putMDC(mdc, MDC_USER_AGENT, request.getHeader("User-Agent"));
    
            // referrer
            putMDC(mdc, MDC_REFERRER, request.getHeader("Referer"));
    
            // cookies
            Cookie[] cookies = request.getCookies();
            List<String> names = emptyList();
    
            if (cookies != null) {
                names = createArrayList(cookies.length);
    
                for (Cookie cookie : cookies) {
                    names.add(cookie.getName());
                    putMDC(mdc, MDC_COOKIE_PREFIX + cookie.getName(), cookie.getValue());
                }
    
                sort(names);
            }
    
            putMDC(mdc, MDC_COOKIES, names.toString());
        }

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

      try {   helper.setLoggingContext();
    
                chain.doFilter(request, response);
            } finally {
                helper.clearLoggingContext();
            }

     

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

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

    配置log4j:

     <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

    java.lang.NullPointerException
    	at com.alibaba.citrus.webx.impl.WebxControllerImpl.service(WebxControllerImpl.java:42)
    	at com.alibaba.citrus.webx.impl.WebxRootControllerImpl.handleRequest(WebxRootControllerImpl.java:53)
    	at com.alibaba.citrus.webx.support.AbstractWebxRootController.service(AbstractWebxRootController.java:156)
    	at com.alibaba.citrus.webx.servlet.WebxFrameworkFilter.doFilter(WebxFrameworkFilter.java:141)
    	at com.alibaba.citrus.webx.servlet.FilterBean.doFilter(FilterBean.java:164)
    	at org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1322)
    	at com.alibaba.citrus.webx.servlet.SetLoggingContextFilter.doFilter(SetLoggingContextFilter.java:62)
    	at com.alibaba.citrus.webx.servlet.FilterBean.doFilter(FilterBean.java:164)
    	at org.eclipse.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1322)


  • 相关阅读:
    复数加法
    通过函数来实现复数相加
    声明一个类模板
    友元成员函数的简单应用
    将普通函数声明为友元函数
    引用静态数据成员
    对象的赋值
    对象的常引用
    有关对象指针的使用方法
    对象数组的使用方法
  • 原文地址:https://www.cnblogs.com/secbook/p/2655170.html
Copyright © 2011-2022 走看看