`
wang_zhi_peng2007
  • 浏览: 248929 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

主题:Log4J实战

 
阅读更多

场景如下:
   由于项目要做一个统计分析与记录的功能,为了后期分析用户的一些行为,同时记录的东西要同时输出多处,并且可配置输出

想到了使用log4j,知道他是可以同时指定多个输出目的地,并且如有变更,直接修改配置文件。

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


        第一步:记录用户访问,写一个filter如下,里面MDC是一个类似map的东西,只不过做了线程方式的封装使用,每个线程会有自己的一份map,可以看这个http://www.iteye.com/topic/1112590

public class UserLogFilter implements Filter {
	private Logger logger;

	@Override
	public void destroy() {
		// TODO Auto-generated method stub
		
	}

	@Override
	public void doFilter(ServletRequest request, ServletResponse response,
			FilterChain chain) throws IOException, ServletException {
		HttpServletRequest httpRequest = (HttpServletRequest) request;
		
		Object user = httpRequest.getSession().getAttribute(
				Constants.User.LOGIN_USER);
		if (user != null) {
			Integer usr_id = (Integer) ReflectionUtils.invokeGetterMethod(user, "usrId");
			String log_ip = request.getLocalAddr();
			SimpleDateFormat format=new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
			Date now=new Date(System.currentTimeMillis());
			MDC.put("usr_id", usr_id);
			MDC.put("log_title", "网站访问记录");
			MDC.put("log_type", "记录");
			MDC.put("log_title", "网站访问记录");
			MDC.put("log_datetime", format.format(now));
			MDC.put("log_ip", log_ip);
			logger.info(MDC.getContext());
		}
		 chain.doFilter(request, response); 
		
	}

	@Override
	public void init(FilterConfig filterConfig) throws ServletException {
		logger=Logger.getLogger(UserLogFilter.class);
		
	}
	
}

              第二步:配置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,一个来存东西,一个来控制清零后默认的缓冲大小不变,具体看源码,配置文件可以直接配置初始化大小

 

<p>Each append call adds to an <code>ArrayList</code> buffer.  When
  the buffer is filled each log event is placed in a sql statement
  (configurable) and executed.

  <b>BufferSize</b>, <b>db URL</b>, <b>User</b>, & <b>Password</b> are
  configurable options in the standard log4j ways.

  <p>The <code>setSql(String sql)</code> sets the SQL statement to be
  used for logging -- this statement is sent to a
  <code>PatternLayout</code> (either created automaticly by the
  appender or added by the user).  Therefore by default all the
  conversion patterns in <code>PatternLayout</code> can be used
  inside of the statement.  (see the test cases for examples)
 protected int bufferSize = 1;

  /**
   * ArrayList holding the buffer of Logging Events.
   */
  protected ArrayList buffer;

  /**
   * Helper object for clearing out the buffer
   */
  protected ArrayList removes;

    里面执行的sql,连接url 什么的就是配置文件里面配置,,会在调用的时候set进来

 

 protected String databaseURL = "jdbc:odbc:myDB";

  /**
   * User to connect as for default connection handling
   */
  protected String databaseUser = "me";

  /**
   * User to use for default connection handling
   */
  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的,是无法获取到,只能自己再来一份

public class DBAppender extends org.apache.log4j.jdbc.JDBCAppender {
	private BoneCP connectionPool = null;
	private Connection connection = null;
	private static Logger logger=Logger.getLogger(DBAppender.class);
	public DBAppender() {
		// 设置连接池配置信息
		BoneCPConfig config = new BoneCPConfig();
		PropetiesUtil p;
	
		try {
			Properties P = new Properties();
			P.load(DBAppender.class.getClassLoader().getResourceAsStream("application.properties"));
			// 数据库的JDBC URL
			config.setJdbcUrl(P.getProperty("jdbc.url"));
			// 数据库用户名
			config.setUsername(P.getProperty("jdbc.username"));
			// 数据库用户密码
			config.setPassword(P.getProperty("jdbc.password"));
			// 数据库连接池的最小连接数
			config.setMinConnectionsPerPartition(5);
			// 数据库连接池的最大连接数
			config.setMaxConnectionsPerPartition(10);
			config.setPartitionCount(1);
			// 设置数据库连接池
			connectionPool = new BoneCP(config);
			
		
		} catch (SQLException e) {
			// TODO Auto-generated catch block
			logger.error("连接池配置加载异常",e);
		} catch (IOException e) {
			// TODO Auto-generated catch block
			logger.error("加载配置文件IO异常",e);
		}

		// fetch a connection

	}

	@Override
	protected Connection getConnection() throws SQLException {
		if(connection==null||connection.isClosed()){
			connection = connectionPool.getConnection();
		}
		return connection;
	}

	@Override
	protected void closeConnection(Connection con) {
		// TODO Auto-generated method stub
		try {
			connection.close();
			connection=null;
		} catch (SQLException e) {
			// TODO Auto-generated catch block
			logger.error("连接没正常关闭",e);
		}
	}

}

    第四步:一切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从任务文件里面分析出数据然后写入数据库于日志文件

  • 大小: 143.7 KB
分享到:
评论

相关推荐

    Log4j将System.out搞到log4j中输出四

    在《Log4j将System.out搞到log4j中输出四》这篇博文中,作者可能详细讨论了这些步骤,并可能分享了一些实战经验。通过学习这篇博文,读者可以更深入地了解如何在实际项目中实现这一转换,提升日志管理的效率。 总结...

    log4j the complete manual

    《log4j完全手册》是Java开发者不可或缺的参考资料,它详细介绍了如何在Java应用程序中使用log4j这一广泛使用的日志记录框架。log4j是由Apache软件基金会开发的一个开源项目,旨在提供灵活、高效和可扩展的日志记录...

    log4j文档及使用

    本文将围绕“log4j文档及使用”这一主题,深入探讨其核心概念、配置方式以及实战应用。 首先,Log4j的核心理念是提供一种灵活的日志记录机制,允许开发者控制日志信息的级别(如DEBUG、INFO、WARN、ERROR和FATAL)...

    Pro Apache Log4j second edition

    - **全面性**:该书涵盖了从基础到高级的所有Log4j相关主题,适合不同层次的学习者。 - **实用性**:书中包含大量实际代码示例和配置文件模板,便于读者快速上手。 - **深入浅出**:作者通过通俗易懂的语言和丰富的...

    Flume + kafka + log4j构建日志采集系统

    本文将详细探讨如何使用Flume、Kafka和log4j构建一个高效的日志采集系统,帮助你理解这三个组件在日志处理中的角色以及如何协同工作。 首先,让我们了解一下这三个工具的基本概念: 1. **Flume**: Apache Flume 是...

    log4j资料(文档、实例)

    四、Log4j实战 在Java代码中,我们可以使用以下方式调用Log4j: ```java import org.apache.log4j.Logger; public class MyClass { private static final Logger logger = Logger.getLogger(MyClass.class); ...

    Web 式样书,Log4J,发送Email

    本实战项目旨在帮助开发者掌握Web应用程序的实现,包括使用MVC设计模式、数据源(DataSource)、数据传输对象(DTO)、数据访问对象(DAO)、电子邮件发送以及日志管理工具Log4J的配置。此外,还将涉及文件上传功能...

    JavaEE开发的颠覆者 Spring Boot实战 完整版.zip

    7. **日志管理**:使用Logback或Log4j2进行日志记录,以及日志的分级和远程日志传输。 8. **微服务实战**:如何利用Spring Cloud构建微服务架构,包括服务发现、配置中心、负载均衡等。 9. **持续集成与部署**:...

    spring实战中需要的jar

    Spring框架在内部大量使用了Commons Logging,以便在运行时选择合适的日志实现,如Log4j或java.util.logging。这样做的好处是,开发者可以根据项目需求轻松切换日志系统,而无需修改代码。 在实际项目中,这些jar...

    2019实战ActiveMQ集群与应用实战视频教程

    - 配置 log4j.properties 文件,以便于日志记录。 - **启动与监控**: - 使用命令行工具启动 ActiveMQ 服务器。 - 通过 Web 控制台(默认端口为 8161)监控集群状态。 - 监控指标包括队列长度、消息速率等。 - *...

    大型商场项目实战springmvc视频教程

    - 使用Log4j或SLF4J记录应用日志,便于问题排查。 3. **监控与性能调优**: - 使用JProfiler等工具对项目进行性能分析,优化瓶颈。 4. **持续集成与自动化测试**: - 使用Jenkins搭建持续集成环境,自动化构建...

    SpringBoot实战教程(亲测非常好)百度网盘链接

    - **Logback** 和 **Log4j2** 是Spring Boot中常用的日志框架。 - **配置示例**:在 **logback-spring.xml** 中配置日志级别。 ```xml &lt;pattern&gt;%d{HH:mm:ss.SSS} [%thread] %-5level %logger{36} - %msg%n ...

    实战突击javaweb项目整合开发项目源代码11-21

    12. **日志记录**:如Log4j或SLF4J用于记录应用程序的运行日志,便于问题排查。 13. **异常处理**:通过try-catch-finally结构处理可能出现的运行时异常,保证程序的健壮性。 14. **性能优化**:包括数据库查询...

    实战突击 JavaWeb 项目整合开发

    同时,使用Log4j或Logback记录日志,有助于调试和问题排查。 7. **测试**:JUnit和Mockito等工具可以帮助进行单元测试和集成测试,确保代码质量和功能的正确性。 8. **部署与运维**:了解Tomcat、Jetty等应用...

    服务器端接口编程.pptx

    本章重点介绍了四个核心主题,分别是JNDI数据源配置、Log4J与SLF4J日志记录、MyBatis与Hibernate ORM框架的比较以及一个实际的项目案例——移动商城。 **1. JNDI 数据源配置** JNDI,全称为Java Naming and ...

    Spring Boot 系列实战合集.zip

    5. 日志:内置了日志处理框架,可以根据环境自动选择合适的日志系统(如 Logback 或 Log4j2)。 6. Actuator:提供了健康检查、审计、指标收集等监控功能,帮助开发者了解应用的运行状态。 7. Starter:预定义的 ...

    java信息管理系统SE实战源码 Java学习资料

    9. **日志记录**:了解如何使用如Log4j等工具进行调试和故障排查。 10. **国际化与本地化**:如果系统支持多语言,会涉及资源文件的管理。 通过深入研究这个实战源码,学习者可以全面提升Java SE开发技能,理解实际...

    SpringBoot入门到项目实战课程相关资料

    6. **日志管理**:学习如何配置和使用Logback或Log4j2进行日志记录。 7. **单元测试**:利用JUnit和Mockito进行单元测试,确保代码质量。 8. **Actuator**:Spring Boot提供的监控和健康检查工具,用于生产环境的...

    Java学习路线图.docx

    16. **日志工具**:Log4j和Slf4j用于记录应用程序的运行日志。 17. **注解**:用于减少配置文件,提高代码可读性和维护性。 18. **源代码版本管理**:SVN和Git是版本控制工具,用于协同开发和版本管理。 **应用...

    hibernate quickly

    - `log4j.properties`:Log4j的日志配置文件,用于控制程序运行时的输出信息级别和目的地,有助于调试和性能分析。 2. **构建文件** - `hibernate-build.xml`:Ant或Maven的构建脚本,用于自动化构建项目,包括...

Global site tag (gtag.js) - Google Analytics