转自:http://blog.csdn.net/zhongweijian/article/details/7625279
分类: 开源框架 java2012-06-02 08:30 393人阅读 收藏 举报
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的实现代码:
- 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的实现:
在webx框架中对于log4j的MDC的处理:
先配置一个filter,这个filter是放置web.xml的最前面
- protected void populateMDC(Map<String, String> mdc) {
-
- 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);
-
-
- putMDC(mdc, MDC_REMOTE_HOST, request.getRemoteHost());
- putMDC(mdc, MDC_REMOTE_ADDR, request.getRemoteAddr());
-
-
- putMDC(mdc, MDC_USER_AGENT, request.getHeader("User-Agent"));
-
-
- putMDC(mdc, MDC_REFERRER, request.getHeader("Referer"));
-
-
- 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:1
分享到:
相关推荐
最新的日志log4j帮助文档 帮你快速玩转log4j
同时,结合日志工具(如Log4j或Logback),可以帮助跟踪认证过程中的信息流,这对于排查问题和理解系统行为非常有用。 **学习与应用**: 深入理解`AuthenticationProcessingFilter`的运作机制对于开发和维护基于...
【玩转QQ空间的技巧大全】 QQ空间作为中国最受欢迎的社交平台之一,拥有众多的使用者。对于新手来说,想要充分利用并个性化自己的空间可能需要一些技巧。以下是一些玩转QQ空间的方法: 1. **购物与摆放挂件**: ...
基于wechat4j开发 使用方式 将conf/wechat4j.properties拷贝到主项目的conf下面,并配置好相关的配置即可。 使用手册 与wechat4j的不同 json采用play自带的gson httpclient部分采用play封装的webservice部分play.lib...
日志管理可能使用Logback或Log4j。 10. **持续集成/持续部署(CI/CD)**:为了确保代码质量和自动化部署,项目可能使用Jenkins、GitLab CI/CD或Travis CI等工具实现持续集成和持续部署。 综上所述,这个项目融合了...
- log4j - tk.mybatis.mapper 特别地,tk.mybatis.mapper是通用mapper的依赖库,版本为3.4.2。 配置好项目后,你可以开始使用通用mapper。例如,创建一个User实体类,然后通过注解和通用mapper提供的接口,就可以...
此外,日志系统(如Log4j)可以帮助开发者追踪程序运行过程中的问题,以便进行调试。 7. **版本控制**:"Java-master"的命名方式暗示了可能使用了Git作为版本控制系统。开发者可以使用Git进行代码的版本管理,协同...
9. **日志管理**:使用Log4j或Spring Boot内置的日志系统进行日志记录。 10. **异常处理**:学习如何在Spring应用中优雅地处理异常。 通过深入研究这个项目,开发者可以从实践中学习到如何平滑地从C#迁移到Java和...
因此,(1,1,5) 将产生序列 1 2 3 4 5,(5,-1,1) 将产生 序列 (5 4 3 2 1)。 FOR /F ["options"] %variable IN (file-set) DO command FOR /F ["options"] %variable IN ("string") DO command FOR /F ["options"]...