log4j实战

原文:http://www.iteye.com/topic/1112588

场景如下: 

   由于项目要做一个统计分析与记录的功能,为了后期分析用户的一些行为,同时记录的东西要同时输出多处,并且可配置输出  
想到了使用log4j,知道他是可以同时指定多个输出目的地,并且如有变更,直接修改配置文件。 

关于log4j的基本概论及使用,见http://www.iteye.com/topic/378077,这个帖子讲的很细,归纳了很多东西 

问题一:把登录用户每次访问的ip地址 时间等记录下来,存放到日志文件里,再存一份到数据库 

第一步:记录用户访问,写一个filter如下,里面MDC是一个类似map的东西,只不过做了线程方式的封装使用,每个线程会有自己的一份map,可以看这个http://www.iteye.com/topic/1112590  
Java代码   收藏代码
  1. public class UserLogFilter implements Filter {  
  2.     private Logger logger;  
  3.   
  4.     @Override  
  5.     public void destroy() {  
  6.         // TODO Auto-generated method stub  
  7.           
  8.     }  
  9.   
  10.     @Override  
  11.     public void doFilter(ServletRequest request, ServletResponse response,  
  12.             FilterChain chain) throws IOException, ServletException {  
  13.         HttpServletRequest httpRequest = (HttpServletRequest) request;  
  14.           
  15.         Object user = httpRequest.getSession().getAttribute(  
  16.                 Constants.User.LOGIN_USER);  
  17.         if (user != null) {  
  18.             Integer usr_id = (Integer) ReflectionUtils.invokeGetterMethod(user, "usrId");  
  19.             String log_ip = request.getLocalAddr();  
  20.             SimpleDateFormat format=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");  
  21.             Date now=new Date(System.currentTimeMillis());  
  22.             MDC.put("usr_id", usr_id);  
  23.             MDC.put("log_title""网站访问记录");  
  24.             MDC.put("log_type""记录");  
  25.             MDC.put("log_title""网站访问记录");  
  26.             MDC.put("log_datetime", format.format(now));  
  27.             MDC.put("log_ip", log_ip);  
  28.             logger.info(MDC.getContext());  
  29.         }  
  30.          chain.doFilter(request, response);   
  31.           
  32.     }  
  33.   
  34.     @Override  
  35.     public void init(FilterConfig filterConfig) throws ServletException {  
  36.         logger=Logger.getLogger(UserLogFilter.class);  
  37.           
  38.     }  
  39.       
  40. }  


第二部:配置log4j.properties,输出到文件与数据库  
引用

#不懂的话,参考给出的第一个连接地址一样,j2EE项目详细控制 
log4j.logger.com.bhaman.yiyaosou.util.web=INFO,,project-util,project-util-db 
#project-util-web file appender 
log4j.appender.project-util=org.apache.log4j.DailyRollingFileAppender 
log4j.appender.project-util.file=${user.home}/logs/project-util.log 
log4j.appender.project-util.layout=org.apache.log4j.PatternLayout 
log4j.appender.project-util.threshold=INFO 
log4j.appender.project-util.layout.conversionPattern=%d [%X{usr_id}/%X{log_ip}/%X{req.id} - %X{entranceMethod} - %X{req.requestURIWithQueryString}] %-5p %c - %m%n 
log4j.appender.project-util-db=com.log4j.service.DBAppender 
log4j.appender.project-util-db.bufferSize=16 
log4j.appender.project-util-db.threshold=INFO 
#此处对应filter里面的MDC里面的键值对,你懂的 
log4j.appender.project-util-db.sql=insert into user_log (usr_id,log_title,log_category,log_type,log_datetime,log_ip) VALUES ('%X{usr_id}','%X{log_title}','%X{log_type}','%X{log_title}','%X{log_datetime}','%X{log_ip}') 


第三部:由于log4j里面给出的输出JDBCappender,是有问题的,首先面对业务需求,他是用JDBC,性能上问题很大,特别是现在这个应用。log4j支持自己写appender 
直接看JDBCAppender的源码
 

如下里面提到有做了缓冲,两个ArrayList,一个来存东西,一个来控制清零后默认的缓冲大小不变,具体看源码,配置文件可以直接配置初始化大小  
Java代码   收藏代码
  1. <p>Each append call adds to an <code>ArrayList</code> buffer.  When  
  2.   the buffer is filled each log event is placed in a sql statement  
  3.   (configurable) and executed.  
  4.   
  5.   <b>BufferSize</b>, <b>db URL</b>, <b>User</b>, & <b>Password</b> are  
  6.   configurable options in the standard log4j ways.  
  7.   
  8.   <p>The <code>setSql(String sql)</code> sets the SQL statement to be  
  9.   used for logging -- this statement is sent to a  
  10.   <code>PatternLayout</code> (either created automaticly by the  
  11.   appender or added by the user).  Therefore by default all the  
  12.   conversion patterns in <code>PatternLayout</code> can be used  
  13.   inside of the statement.  (see the test cases for examples)  
  14.  protected int bufferSize = 1;  
  15.   
  16.   /** 
  17.    * ArrayList holding the buffer of Logging Events. 
  18.    */  
  19.   protected ArrayList buffer;  
  20.   
  21.   /** 
  22.    * Helper object for clearing out the buffer 
  23.    */  
  24.   protected ArrayList removes;  


里面执行的sql,连接url 什么的就是配置文件里面配置,,会在调用的时候set进来 
Java代码   收藏代码
  1. protected String databaseURL = "jdbc:odbc:myDB";  
  2.   
  3.  /** 
  4.   * User to connect as for default connection handling 
  5.   */  
  6.  protected String databaseUser = "me";  
  7.   
  8.  /** 
  9.   * User to use for default connection handling 
  10.   */  
  11.  protected String databasePassword = "mypassword";  


那么我自己写的appender继承它就好了,直接使用它的缓冲及sql执行,那么唯一要变的就是连接了。 
我要从连接池里面取出,怎么做呢?看下面源码中注释-->>
 

引用

   <li>Override <code>getConnection()</code> to pass any connection 
    you want.  Typically this is used to enable application wide 
    connection pooling. 

     <li>Override <code>closeConnection(Connection con)</code> -- if 
     you override getConnection make sure to implement 
     <code>closeConnection</code> to handle the connection you 
     generated.  Typically this would return the connection to the 
     pool it came from. 

     <li>Override <code>getLogStatement(LoggingEvent event)</code> to 
     produce specialized or dynamic statements. The default uses the 
     sql option value. 

我不需要覆写getLogStatement,我想改变的只是连接的获取方式罢了,如上分析后,直接写我的appender,里面用到了高效的BoneCP连接池,在spring里面本来是有配置这个的,但log4j是独立于spring的,是无法获取到,只能自己再来一份  

Java代码   收藏代码
  1. public class DBAppender extends org.apache.log4j.jdbc.JDBCAppender {  
  2.     private BoneCP connectionPool = null;  
  3.     private Connection connection = null;  
  4.     private static Logger logger=Logger.getLogger(DBAppender.class);  
  5.     public DBAppender() {  
  6.         // 设置连接池配置信息  
  7.         BoneCPConfig config = new BoneCPConfig();  
  8.         PropetiesUtil p;  
  9.       
  10.         try {  
  11.             Properties P = new Properties();  
  12.             P.load(DBAppender.class.getClassLoader().getResourceAsStream("application.properties"));  
  13.             // 数据库的JDBC URL  
  14.             config.setJdbcUrl(P.getProperty("jdbc.url"));  
  15.             // 数据库用户名  
  16.             config.setUsername(P.getProperty("jdbc.username"));  
  17.             // 数据库用户密码  
  18.             config.setPassword(P.getProperty("jdbc.password"));  
  19.             // 数据库连接池的最小连接数  
  20.             config.setMinConnectionsPerPartition(5);  
  21.             // 数据库连接池的最大连接数  
  22.             config.setMaxConnectionsPerPartition(10);  
  23.             config.setPartitionCount(1);  
  24.             // 设置数据库连接池  
  25.             connectionPool = new BoneCP(config);  
  26.               
  27.           
  28.         } catch (SQLException e) {  
  29.             // TODO Auto-generated catch block  
  30.             logger.error("连接池配置加载异常",e);  
  31.         } catch (IOException e) {  
  32.             // TODO Auto-generated catch block  
  33.             logger.error("加载配置文件IO异常",e);  
  34.         }  
  35.   
  36.         // fetch a connection  
  37.   
  38.     }  
  39.   
  40.     @Override  
  41.     protected Connection getConnection() throws SQLException {  
  42.         if(connection==null||connection.isClosed()){  
  43.             connection = connectionPool.getConnection();  
  44.         }  
  45.         return connection;  
  46.     }  
  47.   
  48.     @Override  
  49.     protected void closeConnection(Connection con) {  
  50.         // TODO Auto-generated method stub  
  51.         try {  
  52.             connection.close();  
  53.             connection=null;  
  54.         } catch (SQLException e) {  
  55.             // TODO Auto-generated catch block  
  56.             logger.error("连接没正常关闭",e);  
  57.         }  
  58.     }  
  59.   
  60. }  


第四步:一切OK,然后就测试
 
日志文件: 
引用

2011-07-21 11:07:01,125 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:07:01, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:07:03,640 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:07:03, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:07:04,796 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:07:04, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:07:08,906 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:07:08, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:07:09,281 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:07:09, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:07:14,531 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:07:14, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:10,984 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:10, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:11,796 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:11, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:22,078 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:22, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:22,875 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:22, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:28,562 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:28, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:30,250 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:30, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:31,390 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:31, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:32,750 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:32, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:33,781 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:33, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:36,156 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:36, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:41,578 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:41, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:43,156 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:43, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:44,968 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:44, log_title=网站访问记录, usr_id=282} 
2011-07-21 11:11:48,765 [282/127.0.0.1/ -  - ] INFO  com.bhaman.yiyaosou.util.web.UserLogFilter - {log_type=记录, log_ip=127.0.0.1, log_datetime=2011-07-21 11:11:48, log_title=网站访问记录, usr_id=282} 


数据库: 

 

第五步: 
    比如现在系统要记录登录用户与非登录用户各个时间段得访问与访问连接数,最后跟购买行为统计挂上钩,进行数据挖掘,但现在又改变需求,还要分析登录用户里面的各种级别的用户等 
    系统里面需要统计分析的东西很多的话,怎么解决?难道配置各种filter然后输出两地?本来太多的fiter就会带来性能问题,多了岂不是? 
    大范围记录一次到日志文件,然后写job从任务文件里面分析出数据然后写入数据库于日志文件。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值