- 浏览: 755125 次
- 性别:
- 来自: 杭州
文章分类
最新评论
-
lgh1992314:
a offset: 26b offset: 24c offse ...
java jvm字节占用空间分析 -
ls0609:
语音实现在线听书http://blog.csdn.net/ls ...
Android 语音输入API使用 -
wangli61289:
http://viralpatel-net-tutorials ...
Android 语音输入API使用 -
zxjlwt:
学习了素人派http://surenpi.com
velocity宏加载顺序 -
tt5753:
谢啦........
Lucene的IndexWriter初始化时的LockObtainFailedException的解决方法
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的实现:
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)
发表评论
-
对字符串进行验证之前先进行规范化
2013-09-17 23:18 13957对字符串进行验证之前先进行规范化 应用系统中经常对字 ... -
使用telnet连接到基于spring的应用上执行容器中的bean的任意方法
2013-08-08 09:17 1482使用telnet连接到基于spring的应用上执行容器中 ... -
jdk7和8的一些新特性介绍
2013-07-06 16:07 10114更多ppt内容请查看:htt ... -
Lucene的IndexWriter初始化时的LockObtainFailedException的解决方法
2013-06-28 21:35 11808原文链接: http://www.javaarch.net ... -
java对于接口和抽象类的代理实现,不需要有具体实现类
2013-06-12 09:50 2957原文链接:http://www.javaarch.net/j ... -
Excel2007格式分析和XML解析
2013-06-07 09:56 10745在物料清单采购中,用到excel上传文件解析功能,不 ... -
Java EE 7中对WebSocket 1.0的支持
2013-06-05 09:27 3846原文链接:http://www.javaarch.n ... -
java QRCode生成示例
2013-06-05 09:26 1515原文链接:http://www.javaarch.n ... -
Spring Security Logout
2013-06-03 00:05 2371原文地址:http://www.javaarch.net/ ... -
Spring Security Basic Authentication
2013-06-03 00:04 1742原文地址:http://www.javaarch.net/ ... -
Spring Security Form Login
2013-06-02 16:16 2149原文地址:http://www.javaarch.net/j ... -
spring3 的restful API RequestMapping介绍
2013-06-02 14:53 1155原文链接:http://www.javaarch.net/j ... -
Java Web使用swfobject调用flex图表
2013-05-28 19:05 1128Java Web使用swfobject调用 ... -
spring使用PropertyPlaceholderConfigurer扩展来满足不同环境的参数配置
2013-05-21 15:57 3342spring使用PropertyPlaceholderCon ... -
java国际化
2013-05-20 20:57 4478java国际化 本文来自:http://www.j ... -
RSS feeds with Java
2013-05-20 20:52 1226RSS feeds with Java 原文来自:htt ... -
使用ibatis将数据库从oracle迁移到mysql的几个修改点
2013-04-29 10:40 1679我们项目在公司的大战略下需要从oracle ... -
线上机器jvm dump分析脚本
2013-04-19 10:48 2911#!/bin/sh DUMP_PIDS=`p ... -
spring3学习入门示例工程
2013-04-18 09:28 11331. github地址 https://github ... -
spring map使用annotation泛型注入问题分析
2013-04-15 13:30 8547今天在整一个spring的ioc学习demo,碰到 ...
相关推荐
Log4j是Apache软件基金会开发的一个开源项目,它为Java应用程序提供了一个灵活的日志系统,允许开发者自定义日志级别、输出格式以及存储位置等,以满足不同场景的需求。 描述中的"NULL"意味着没有额外的具体信息,...
以上配置覆盖了Log4J常用的几种Appender类型及其配置参数,能够满足大多数应用场景的需求。开发者可以根据实际需求调整各个Appender的配置,以实现更加精细化的日志管理。此外,Log4J还提供了丰富的扩展机制,如MDC...
总的来说,Apache Log4j 2是一个强大且灵活的日志框架,它不仅提供了高效的日志记录能力,还通过丰富的配置选项和插件支持,满足了各种复杂应用场景的需求。在实际项目中,你可以根据“apache-log4j-2.4.1-bin.zip”...
5. **MDC与NDC**:Log4j-SLF4J-Impl支持Mapped Diagnostic Context (MDC) 和Nested Diagnostic Context (NDC),为日志信息添加上下文信息,有助于追踪和诊断问题。 四、集成与使用 集成Log4j-SLF4J-Impl通常涉及...
Apache Log4j 是一个广泛使用的开源日志记录框架,为Java应用程序提供了强大的日志处理能力。在2.11.1这个版本中,Log4j 继续提供高效、灵活的日志记录解决方案,旨在满足各种开发需求。本文将深入探讨Log4j 2.11.1...
对于高并发场景,Log4j还提供了异步日志功能,通过使用AsyncAppender,可以避免日志输出对主线程的影响,提高应用性能。 7. **MDC与NDC** 异常跟踪时,可以使用Mapped Diagnostic Context (MDC) 和 Nested ...
总结来说,Log4j是一个强大且灵活的日志框架,它的广泛使用证明了其在Java应用日志管理中的价值。通过深入理解和正确配置,Log4j可以帮助开发者更好地监控程序状态,定位和解决问题,从而提升软件的稳定性和可靠性。...
Log4j是Apache组织开发的一款广泛使用的Java日志框架,主要功能是用于记录应用程序运行过程中的各种日志信息。在Java编程中,日志记录是非常重要的一环,它可以帮助开发者追踪程序运行状态,定位错误,优化性能,...
**Log4j2使用案例** 在Java开发中,日志记录是不可或缺的一部分,它帮助开发者追踪程序运行状态、定位问题和进行性能分析。Log4j2作为Apache的一个顶级项目,是Log4j的升级版,提供了更高效、更灵活的日志处理能力...
LOG4J还支持MDC(Mapped Diagnostic Context)和NDC(Nested Diagnostic Context),它们提供了上下文信息,帮助追踪特定请求或线程的日志。此外,LOG4J的异步日志记录功能可以提高日志处理效率,尤其在高并发场景下...
- **NDC与MDC**:NDC和MDC是Log4j提供的两个重要的上下文管理机制。NDC用于在多层调用中保持调用链的上下文信息,MDC则用于存储线程特有的键值对数据,便于日志分析。 - **动态配置**:Log4j支持在运行时动态修改...
除了基本的日志功能,log4j-1.2.15还支持一些高级特性,如异步日志记录(使用AsyncAppender)、日志事件的自定义处理(通过编写自定义Appender和Layout)、MDC(Mapped Diagnostic Context)和NDC(Nested ...
本示例包含Log4j的使用说明及对应的jar包,旨在帮助开发者快速理解和应用Log4j。 **一、Log4j的基本组件** 1. **Logger**: 日志记录器,是日志系统的核心,负责生成日志信息。每个类通常都有一个与之关联的Logger...
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....
10. **使用场景**:Log4j适用于各种Java应用,尤其在调试、故障排查、性能监控等方面。 **源代码分析** 在提供的压缩包中,可能包含了示例代码,这些代码展示了如何在Java项目中集成和使用Log4j。通过分析这些代码...
在使用Hibernate这样的ORM框架时,Log4j经常被用来记录应用的运行状态和调试信息。 **1. 日志框架的重要性** 日志是软件开发中不可或缺的一部分,它帮助开发者追踪程序运行时的问题,进行性能分析,以及优化代码。...
Log4j是Apache软件基金会开发的一个开源日志组件,它为Java应用程序提供了灵活的日志记录功能。Log4j 1.2.12是其1.2系列的一个稳定版本,包含了一系列优化和改进,旨在提高性能和稳定性。 2. **核心组件** - **...
在Java编程领域,日志记录是不可或缺的一部分,而Log4j是一个广泛使用的开源日志框架,它为应用程序提供了一种灵活的日志记录方式。这个压缩包文件“log4jTest”很可能包含了与Log4j相关的测试代码或示例,帮助我们...
Log4j 是一款广泛使用的Java日志记录框架,它的主要目标是为应用程序提供灵活的日志输出控制,方便调试和监控。以下是对Log4j及其相关工具的详细解释: 1. **日志介绍**: 日志是记录软件运行时状态的重要手段,它...