Log4j写入数据库详解

Posted on

Log4j写入数据库详解 - ziruobing的专栏 - CSDN博客

您还未登录!|登录|注册|帮助

ziruobing的专栏

黑暗中漫舞

*

公告:

原创 Log4j写入数据库详解 收藏

log4j是一个优秀的开源日志记录项目,我们不仅可以对输出的日志的格式自定义,还可以自己定义日志输出的目的地,比如:屏幕,文本文件,数据库,甚至能通过socket输出。本节主要讲述如何将日志信息输入到数据库(可以插入任何数据库,在此主要以MSSQL为例进行详解)。 用log4j将日志写入数据库主要用到是log4j包下的JDBCAppender类,它提供了将日志信息异步写入数据的功能,我们可以直接使用这个类将我们的日志信息写入数据库;也可以扩展JDBCAppender类,就是将JDBCAppender类作为基类。下面将通过一个实例来讲解log4j是如何将日志信息写入数据库的。 我们的需求:我们在软件开发的过程中需要将调试信息、操作信息等记录下来,以便后面的审计,这些日志信息包括用户ID、用户姓名、操作类、路径、方法、操作时间、日志信息。 设计思想:我们采用JDBCAppender类直接将日志信息插入数据库,所有只需要在配置文件配置此类就可以;要获得用户信息需要用过滤器来实现;(假如不需要用户的信息,就不需要设计过滤器,其实大部分情况下都是需要这些用户信息,尤其是在web应用开发中)在日志信息中获得用户信息,就的通过过滤器的request或session对象,从session中拿到用户信息怎样传到log4j呢,log4j为我们提供了MDC(MDC是log4j种非常有用类,它们用于存储应用程序的上下文信息(context infomation),从而便于在log中使用这些上下文信息。MDC内部使用了类似map的机制来存储信息,上下文信息也是每个线程独立地储存,所不同的是信息都是以它们的key值存储在”map”中。相对应的方法,

MDC.put(key, value); MDC.remove(key); MDC.get(key);

在配置PatternLayout的时候使用:%x{key}来输出对应的value)。有了MDC,我们可以在过滤器中先获得用户信息,再用MDC.Put(“key”)方法,log在执行sql语句时通过%x{key}来输出对应的value。

实现步骤: 1、在你的项目中要确保有log4j和commons-logging这两个jar文件; 2、设置要你要插入日志信息的表结构

  1. if exists (select /* from dbo.sysobjects where id = object_id(N'[dbo].[WDZLOG]') and OBJECTPROPERTY(id, N'IsUserTable') = 1)
  2. drop table [dbo].[WDZLOG]
  3. GO
  4. CREATE TABLE [dbo].[WDZLOG] (
  5. [WDZLOGID] [int] IDENTITY (1, 1) NOT NULL ,
  6. [LogName] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//用户ID
  7. [UserName] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//用户姓名
  8. [Class] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//类名
  9. [Mothod] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL //,方法名
  10. [CreateTime] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//产生时间
  11. [LogLevel] [varchar] (20) COLLATE Chinese_PRC_CI_AS NULL ,//日志级别
  12. [MSG] [varchar] (555) COLLATE Chinese_PRC_CI_AS NULL //日志信息
  13. ) ON [PRIMARY]
  14. GO
    if exists (select /* from dbo.sysobjects where id = object_id(N'[dbo].[WDZLOG]') and OBJECTPROPERTY(id, N'IsUserTable') = 1) drop table [dbo].[WDZLOG] GO CREATE TABLE [dbo].[WDZLOG] ( [WDZLOGID] [int] IDENTITY (1, 1) NOT NULL , [LogName] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//用户ID [UserName] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//用户姓名 [Class] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//类名 [Mothod] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL //,方法名 [CreateTime] [varchar] (255) COLLATE Chinese_PRC_CI_AS NULL ,//产生时间 [LogLevel] [varchar] (20) COLLATE Chinese_PRC_CI_AS NULL ,//日志级别 [MSG] [varchar] (555) COLLATE Chinese_PRC_CI_AS NULL //日志信息 ) ON [PRIMARY] GO

3、配置文件(摘自我们的项目)后面将对此配置文件进行详细讲解,它也log4j的核心部分。

  1. log4j.properties
  2. log4j.rootLogger=INFO,stdout
  3. log4j.logger.org.springframework.web.servlet=INFO,db
  4. log4j.logger.org.springframework.beans.factory.xml=INFO
  5. log4j.logger.com.neam.stum.user=INFO,db
  6. log4j.appender.stdout=org.apache.log4j.ConsoleAppender
  7. log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
  8. log4j.appender.stdout.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} %p [%c] - - <%m>%n
  9. log4j.appender.logfile=org.apache.log4j.DailyRollingFileAppender
  10. log4j.appender.logfile.File=${webapp.root}/WEB-INF/logs/exppower.log
  11. log4j.appender.logfile.DatePattern=.yyyy-MM-dd
  12. log4j.appender.logfile.layout=org.apache.log4j.PatternLayout
  13. log4j.appender.logfile.layout.ConversionPattern=%d %p [%c] wang- <%m>%n
  14. /#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#
  15. /# JDBC Appender
  16. /#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#
  17. /#log4j.logger.business=INFO,db
  18. /#log4j.appender.db=com.neam.commons.MyJDBCAppender
  19. log4j.appender.db=JDBCExtAppender
  20. log4j.appender.db.BufferSize=10
  21. log4j.appender.db.sqlname=log
  22. log4j.appender.db.driver=net.sourceforge.jtds.jdbc.Driver
  23. log4j.appender.db.URL=jdbc:jtds:SqlServer://localhost:1433;DatabaseName=pubs
  24. log4j.appender.db.user=sa
  25. log4j.appender.db.password=sa
  26. log4j.appender.db.sql=insert into WDZLOG (LogName,UserName,Class,Mothod,createTime,LogLevel,MSG) values ('%X{userId}','%X{userName}','%C','%M','%d{yyyy-MM-dd HH:mm:ss}','%p','%m')
  27. log4j.appender.db.layout=org.apache.log4j.PatternLayout
    log4j.properties log4j.rootLogger=INFO,stdout log4j.logger.org.springframework.web.servlet=INFO,db log4j.logger.org.springframework.beans.factory.xml=INFO log4j.logger.com.neam.stum.user=INFO,db log4j.appender.stdout=org.apache.log4j.ConsoleAppender log4j.appender.stdout.layout=org.apache.log4j.PatternLayout log4j.appender.stdout.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} %p [%c] - - <%m>%n log4j.appender.logfile=org.apache.log4j.DailyRollingFileAppender log4j.appender.logfile.File=${webapp.root}/WEB-INF/logs/exppower.log log4j.appender.logfile.DatePattern=.yyyy-MM-dd log4j.appender.logfile.layout=org.apache.log4j.PatternLayout log4j.appender.logfile.layout.ConversionPattern=%d %p [%c] wang- <%m>%n /#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/# /# JDBC Appender /#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/# /#log4j.logger.business=INFO,db /#log4j.appender.db=com.neam.commons.MyJDBCAppender log4j.appender.db=JDBCExtAppender log4j.appender.db.BufferSize=10 log4j.appender.db.sqlname=log log4j.appender.db.driver=net.sourceforge.jtds.jdbc.Driver log4j.appender.db.URL=jdbc:jtds:SqlServer://localhost:1433;DatabaseName=pubs log4j.appender.db.user=sa log4j.appender.db.password=sa log4j.appender.db.sql=insert into WDZLOG (LogName,UserName,Class,Mothod,createTime,LogLevel,MSG) values ('%X{userId}','%X{userName}','%C','%M','%d{yyyy-MM-dd HH:mm:ss}','%p','%m') log4j.appender.db.layout=org.apache.log4j.PatternLayout

4、编写过滤器(ResFilter.java)

  1. import java.io.IOException;
  2. import javax.servlet.Filter;
  3. import javax.servlet.FilterChain;
  4. import javax.servlet.FilterConfig;
  5. import javax.servlet.ServletException;
  6. import javax.servlet.ServletRequest;
  7. import javax.servlet.ServletResponse;
  8. import javax.servlet.http.HttpServletRequest;
  9. import javax.servlet.http.HttpSession;
  10. import org.apache.log4j.Logger;
  11. import org.apache.log4j.MDC;
  12. import com.neam.domain.User;
  13. public class ResFilter implements Filter{
  14. private final static double DEFAULT_USERID= Math.random()/*100000.0;
  15. public void destroy() {
  16. }
  17. public void doFilter(ServletRequest request, ServletResponse response,
  18. FilterChain chain) throws IOException, ServletException {
  19. HttpServletRequest req=(HttpServletRequest)request;
  20. HttpSession session= req.getSession();
  21. if (session==null){
  22. MDC.put("userId",DEFAULT_USERID);
  23. }
  24. else{
  25. User customer=(User)session.getAttribute("user");
  26. if (customer==null){
  27. MDC.put("userId",DEFAULT_USERID);
  28. MDC.put("userName",DEFAULT_USERID);
  29. }
  30. else
  31. {
  32. MDC.put("userId",customer.getName());
  33. MDC.put("userName",customer.getName());
  34. }
  35. }
  36. //logger.info("test for MDC.");
  37. chain.doFilter(request, response);
  38. }
  39. public void init(FilterConfig Config) throws ServletException {
  40. // this.filterConfig = Config;
  41. // String ccc = Config.getServletContext().getInitParameter("cherset");
  42. // this.targetEncoding = Config.getInitParameter("cherset");
  43. }
  44. }
    import java.io.IOException; import javax.servlet.Filter; import javax.servlet.FilterChain; import javax.servlet.FilterConfig; import javax.servlet.ServletException; import javax.servlet.ServletRequest; import javax.servlet.ServletResponse; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpSession; import org.apache.log4j.Logger; import org.apache.log4j.MDC; import com.neam.domain.User; public class ResFilter implements Filter{ private final static double DEFAULT_USERID= Math.random()/*100000.0; public void destroy() { } public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { HttpServletRequest req=(HttpServletRequest)request; HttpSession session= req.getSession(); if (session==null){ MDC.put("userId",DEFAULT_USERID); } else{ User customer=(User)session.getAttribute("user"); if (customer==null){ MDC.put("userId",DEFAULT_USERID); MDC.put("userName",DEFAULT_USERID); } else { MDC.put("userId",customer.getName()); MDC.put("userName",customer.getName()); } } //logger.info("test for MDC."); chain.doFilter(request, response); } public void init(FilterConfig Config) throws ServletException { // this.filterConfig = Config; // String ccc = Config.getServletContext().getInitParameter("cherset"); // this.targetEncoding = Config.getInitParameter("cherset"); } }

5、在需要写入日志的地方引入

  1. private Log logger = LogFactory.getLog(this.getClass());
  2. 在具体方法中就可以写入日志
  3. logger.info("");
  4. logger.debug("");
  5. logger.warn("");
  6. logger.error("");
    private Log logger = LogFactory.getLog(this.getClass()); 在具体方法中就可以写入日志 logger.info(""); logger.debug(""); logger.warn(""); logger.error("");

配置文件详解: log4j.properties log4j.properties log4j.rootLogger=INFO,stdout

//配置根Logger,其语法为: log4j.rootLogger = [ level ] , appenderName1, appenderName2, … level : 是日志记录的优先级,分为OFF、FATAL、ERROR、WARN、INFO、DEBUG、ALL或者您定义的级别。Log4j建议只使用四个级别,优先级从高到低分别是ERROR、WARN、INFO、DEBUG。通过在这里定义的级别,您可以控制到应用程序中相应级别的日志信息的开关。比如在这里定义了INFO级别,则应用程序中所有DEBUG级别的日志信息将不被打印出来。 appenderName:就是指定日志信息输出到哪个地方。您可以同时指定多个输出目的地。 例如:log4j.rootLogger=info,A1,B2,C3 配置了3个输出地方我们可以设置让A1在控制台输出;B2生产日志文件;C3让日志信息插入数据库中。 本例中是将所有的日志信息在控制台打印出来。 log4j.logger.org.springframework.web.servlet=INFO,db //设置将spring下包的某些类的日志信息写入数据库中,并且在控制台上打印出来。(是通过log4j.rootLogger=INFO,stdout来体现的)db是将日志信息写入数据库中 log4j.logger.org.springframework.beans.factory.xml=INFO //本实例中为了让某些包下的日志信息能写入数据库 log4j.logger.com.neam.stum.user=INFO,db //设置自己某个模块下的日志信息既在控制台上打印而且往数据库中保存 //下面是配置在控制台上打印日志信息,在这里就不再仔细描述了 log4j.appender.stdout=org.apache.log4j.ConsoleAppender log4j.appender.stdout.layout=org.apache.log4j.PatternLayout log4j.appender.stdout.layout.ConversionPattern=%d{yyyy-MM-dd HH:mm:ss} %p [%c] - - <%m>%n //下面是配置将日志信息写入文件中,在这里也就不再仔细描述了 log4j.appender.logfile=org.apache.log4j.DailyRollingFileAppender log4j.appender.logfile.File=${webapp.root}/WEB-INF/logs/exppower.log log4j.appender.logfile.DatePattern=.yyyy-MM-dd log4j.appender.logfile.layout=org.apache.log4j.PatternLayout log4j.appender.logfile.layout.ConversionPattern=%d %p [%c] wang- <%m>%n /#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/# /# JDBC Appender /#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/#/# /#log4j.appender.db=com.neam.commons.MyJDBCAppender //下面是配置将日志信息插入数据库, log4j.appender.db=org.apache.log4j.jdbc.JDBCAppender //配置输出目标为数据库(假如要将日志在控制台输出,配置为log4j.appender. stdout =org.apache.log4j.ConsoleAppender;将日志写入文件,配置为log4j.appender.logfile=org.apache.log4j.DailyRollingFileAppender 这样的配置在许多地方都要有,需要可查有关资料),当然你也可以自己扩展org.apache.log4j.jdbc.JDBCAppender这个类,只需要在这里配置就可以了例如我们配置我自己扩展的MyJDBCAppender,配置为/#log4j.appender.db=com.neam.commons.MyJDBCAppender log4j.appender.db.BufferSize=10 //设置缓存大小,就是当有10条日志信息是才忘数据库插一次 log4j.appender.db.driver=net.sourceforge.jtds.jdbc.Driver //设置要将日志插入到数据库的驱动
log4j.appender.db.URL=jdbc:jtds:SqlServer://localhost:1433;DatabaseName=pubs log4j.appender.db.user=sa log4j.appender.db.password=sa log4j.appender.db.sql=insert into WDZLOG (LogName,UserName,Class,Mothod,createTime,LogLevel,MSG) values ('%X{userId}','%X{userName}','%C','%M','%d{yyyy-MM-dd HH:mm:ss}','%p','%m') //设置要插入日志信息的格式和内容,%X{userId}是置取MDC中的key值,因为我们在过滤器中是将用户id和用户姓名放入MDC中,所有在这里可以用%X{userId}和%X{userName}取出用户的ID和用户姓名;'%C'表示日志信息是来自于那个类;%M表示日志信息来自于那个方法中;%d{yyyy-MM-dd HH:mm:ss}表示日志信息产生的时间,{yyyy-MM-dd HH:mm:ss}表示一种时间格式,你也可以直接写成%d;%p表示日志信息的级别(debug info warn error); %m表示你写入的日志信息 log4j.appender.db.layout=org.apache.log4j.PatternLayout

发表于 @ 2009年02月22日 01:05:00 | 评论( 5 )| 编辑| 举报| 收藏

旧一篇:正则表达式入门 | 新一篇:Hibernate的两种事务管理jdbc 和jta方式

-

查看最新精华文章 请访问博客首页相关文章 SQL计算表达式用SQL向/*.txt中追加文本儲存過程萬能分頁利用SQL移动硬盘文件改变SQLServer 数据库所有对象的所有者成dbolog4j把日志写入数据库详解

wofile 发表于Wed Sep 30 2009 12:13:26 GMT+0800 (China Standard Time) IP:举报回复删除tai hao le .... xie xie a ~~~luchangbin_66 发表于Thu Nov 19 2009 10:53:29 GMT+0800 (China Standard Time) IP:举报回复删除Filter在web.xml中过滤那些。。。。。liang_gdong 发表于Mon Nov 30 2009 16:06:18 GMT+0800 (China Standard Time) IP:举报回复删除好东西,支持mwj0103 发表于Wed Jan 06 2010 13:24:41 GMT+0800 (China Standard Time) IP:举报回复删除太让人感动了!~iceyrain 发表于Tue Jan 12 2010 10:22:37 GMT+0800 (China Standard Time) IP:举报回复删除很牛, 受教了 谢谢

专区推荐内容

热门招聘职位 【更多

希望本站内容对您有点用处,有什么疑问或建议请在后面留言评论
转载请注明作者(RobinChia)和出处 It so life ,请勿用于任何商业用途