初入软件开发这一行的人,可能对日志管理的概念并不是很明确,大概是由于经验所限,以至于根本还考虑不到这个问题。
而从某种意义上来说,日志管理实际上也不需要初入这一行的人来管,他们只需要负责实现自己的主要业务逻辑和功能就好了。
我当初刚入行的时候就有很长一段时间完全不用去关心日志,到后来偶尔涉及到的时候,也都是从其他地方采用cv大法直接搬用。
不过,随着工作时间的变化,随着手头上任务重要程度的变化,也随着接触到的项目数量的变化,让我越来越意识到日志的重要性,它在整个系统中发挥着至关重要的作用!
尤其是涉及到需要后期维护的项目,更是经常需要依靠日志来定位问题,可以说他是运行中的项目出问题时,找问题最好的手段。
java中日志管理的技术有很多,像java自身的java.util.logging,apache的commons-logging,以及slf4j、log4j、logback等等。
其中java.util.logging在日常开发中用的不是很多,用的比较多的后边四个,commons-logging和slf4j是接口,log4j和logback是具体的实现,在我所接触的项目中就用到了这几个。
因为java推荐的就是面向接口编程,所以一般推荐使用的就是那两个接口,但是又由于commons-logging的动态绑定造成了一些问题,因此这两个里边又推荐使用slf4j。
同样的,在两种实现中,logback和log4j是由同一个作者开发,logback出现的更晚,更好,因为也就更推荐用logback。
那么综上而言,目前最推荐的java中的日志管理,就是使用slf4j+logback。
实际上,说了这么多,真正用起来是很简单的,只需要导入相关jar包,写好相关配置,然后需要的地方调用就好了,学习的过程中为了比较不同,我也写了一个简单的额例子。
因为目前大部分的项目都是maven管理,spring框架,所以这个例子中也算是顺便联系spring的最基础配置,就也用了spring。
maven的导包配置pom.xml如下,为了比较这四项技术,所以相关的包我全都导了进来,commons-logging是其他jar依赖的,所以便没有手动再导一次:

<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd"><modelVersion>4.0.0</modelVersion><groupId>logTest</groupId><artifactId>logTest</artifactId><packaging>war</packaging><version>0.0.1-SNAPSHOT</version><name>logTest Maven Webapp</name><url>http://maven.apache.org</url><dependencies><dependency><groupId>junit</groupId><artifactId>junit</artifactId><version>4.12</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-test</artifactId><version>4.0.3.RELEASE</version></dependency><dependency><groupId>org.springframework</groupId><artifactId>spring-context</artifactId><version>4.0.3.RELEASE</version></dependency><dependency><groupId>org.slf4j</groupId><artifactId>slf4j-api</artifactId><version>1.7.12</version></dependency><dependency><groupId>ch.qos.logback</groupId><artifactId>logback-core</artifactId><version>1.1.2</version></dependency><dependency><groupId>ch.qos.logback</groupId><artifactId>logback-classic</artifactId><version>1.1.2</version></dependency><dependency><groupId>org.slf4j</groupId><artifactId>slf4j-log4j12</artifactId><version>1.7.7</version></dependency><dependency><groupId>org.slf4j</groupId><artifactId>jcl-over-slf4j</artifactId><version>1.7.12</version></dependency></dependencies><build><finalName>logTest</finalName></build>
</project>

然后写了简单的spring.xml配置:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"xmlns:context="http://www.springframework.org/schema/context"xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-3.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-3.0.xsd"  ><!-- 引入属性文件 --><!-- --> <context:property-placeholder location="classpath:log4j.properties" ignore-unresolvable="true" /><!-- 配置spring注解扫描 --><context:component-scan base-package="logService.service.impl" />
</beans>

很简单,就是配置引入配置文件和spring装配扫描路径,然后是两个不同的日志配置文件,从命名就很容易知道哪个对应的是哪个,首先是log4j.properties:

`log4j.properties:
log4j.rootLogger=WARN, CONSOLE, FILE
## for console
log4j.appender.CONSOLE=org.apache.log4j.ConsoleAppender
log4j.appender.CONSOLE.layout=org.apache.log4j.PatternLayout
log4j.appender.CONSOLE.layout.ConversionPattern=%d{MM-ddHH:mm:ss}[%c-%L][%t][%-4r] - %m%n
log4j.appender.CONSOLE.Encoding=utf-8
## for file
log4j.appender.FILE=org.apache.log4j.RollingFileAppender
log4j.appender.FILE.File=C:/Users/Think/Desktop/log4j.log
log4j.appender.FILE.MaxFileSize=1MB
log4j.appender.FILE.Append = true
log4j.appender.FILE.layout=org.apache.log4j.PatternLayout
log4j.appender.FILE.layout.ConversionPattern=%d{yyyy-MM-ddHH\:mm\:ss} [%t] %-5p  %-4r %x - %m%n
log4j.appender.FILE.Encoding=utf-8`

然后是logback.xml:

<?xml version="1.0" encoding="UTF-8"?>
<configuration debug="false">
<!--定义日志文件的存储地址 勿在 LogBack 的配置中使用相对路径-->
<property name="LOG_HOME" value="C:/Users/Think/Desktop" />
<!-- 控制台输出 -->
<appender name="STDOUT" class="ch.qos.logback.core.ConsoleAppender">
<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
<!--格式化输出:%d表示日期,%thread表示线程名,%-5level:级别从左显示5个字符宽度%msg:日志消息,%n是换行符-->
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
</encoder>
</appender>
<!-- 按照每天生成日志文件 -->
<appender name="FILE" class="ch.qos.logback.core.rolling.RollingFileAppender">
<rollingPolicy class="ch.qos.logback.core.rolling.TimeBasedRollingPolicy">
<!--日志文件输出的文件名-->
<FileNamePattern>${LOG_HOME}/logback.%d{yyyy-MM-dd}.log</FileNamePattern>
<!--日志文件保留天数-->
<MaxHistory>30</MaxHistory>
</rollingPolicy>
<encoder class="ch.qos.logback.classic.encoder.PatternLayoutEncoder">
<!--格式化输出:%d表示日期,%thread表示线程名,%-5level:级别从左显示5个字符宽度%msg:日志消息,%n是换行符-->
<pattern>%d{yyyy-MM-dd HH:mm:ss.SSS} [%thread] %-5level %logger{50} - %msg%n</pattern>
</encoder>
<!--日志文件最大的大小-->
<triggeringPolicy class="ch.qos.logback.core.rolling.SizeBasedTriggeringPolicy">
<MaxFileSize>10MB</MaxFileSize>
</triggeringPolicy>
</appender><!-- 日志输出级别 -->
<root level="WARN">
<appender-ref ref="STDOUT" />
<appender-ref ref="FILE"/>
</root>
</configuration>

这些配置基本上都是最最基础的配置,具体的代表什么意思,网上也有很多很多的说明。
然后分别写了两个使用了common-logging和slf4j的接口:

package logService.service;/*** 使用common-logger* * @author tuzongxun* @date 2017年2月20日 下午3:36:19*/
public interface CommonLogService {public void printLog(String msg);}
package logService.service;/**** Slf4j日志打印* * @author tuzongxun* * @date 2017年2月20日 下午3:39:11*/
public interface Slf4jLogService {public void printLog(String msg);
}

还有对应的实现类,实现类实际上什么都没做,就是调用日志接口随便打印一条日志而已:

package logService.service.impl;import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;
import org.springframework.stereotype.Component;import logService.service.CommonLogService;/*** 使用commons-logger和log4j打印日志* * @author tuzongxun* @date 2017年2月20日 下午3:36:33*/
@Component
public class CommonLogServiceImp implements CommonLogService {private Log logger = LogFactory.getLog(CommonLogServiceImp.class);public void printLog(String msg) {logger.warn("Commonlogger日志打印:" + msg);}
}
package logService.service.impl;import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.stereotype.Component;import logService.service.Slf4jLogService;/**** Slf4j日志打印* * @author tuzongxun* @date 2017年2月20日 下午3:38:55*/
@Component
public class Slf4jLogServiceImpl implements Slf4jLogService {private Logger log = LoggerFactory.getLogger(Slf4jLogServiceImpl.class);@Overridepublic void printLog(String msg) {log.warn("slf4j日志打印:" + msg);}
}

从上边的代码中,实际上根本看不出什么问题,只看到调用了两个接口而已,方法几乎都是一模一样,至于具体用了哪个实现,有什么区别呢,完全不知道,所以我写了对应的test类:

package logTest;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
import logService.service.CommonLogService;
import logService.service.Slf4jLogService;/*** 日志管理测试* * @author tuzongxun* @date 2017年2月22日 下午3:40:17*/
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = "classpath:spring.xml")
public class LogTest {@Autowiredprivate CommonLogService commonLogService;@Autowiredprivate Slf4jLogService slf4jLogService;@Testpublic void commonLogTest() {commonLogService.printLog("commonlog日志打印");}@Testpublic void slf4jLogTest() {slf4jLogService.printLog("slf4j日志打印");}}

那么现在有了代码,我就可以说一说区别了,实际上一开始我说的pom.xml并不是一次导入的,可能有的项目中只有其中几个,而有的项目中我刚才导入的jar包他们也全都导入了。
经过我的测试发现,当使用common-logging的时候,是只能使用log4j的,如果去掉log4j的jar包,那么结果就是运行junit后没有生成对应的日志文件。
而如果用slf4j就可以使用两个实现,只不过和common-logging不同的是,使用slf4j时除开log4j的包,还需要slf4j连接log4j的包。
使用slf4j和logback要导入logback的包自然就不必说了,但是同时到如log4j和logback的包就导致了另一个问题存在,就是使用slf4j的时候不仅会用log4j,还会用logback,导致结果每次都会有两份日志文件。
因此呢,在这种情况下就需要导入另一个包,也就是我导入的

<dependency><groupId>org.slf4j</groupId><artifactId>jcl-over-slf4j</artifactId><version>1.7.12</version></dependency>

有了这个包以后,便不会再有log4j的日志文件出现。
本例子已经上传到csdn下载:
http://download.csdn.net/user/tuzongxun

java程序日志管理相关推荐

  1. 使用comm在java程序中管理本地端口[回钦波:高级软件工程师]

    使用comm在java程序中管理本地端口 最近在做电信项目,遇到一些专业性的问题,在这里和大家一起分享下comm包下的类及如何使用等问题: 在java程序中,为了实现同本地系统的硬件端口(RS-232 ...

  2. 应用程序日志管理工具

    应用程序负责机构中的大量终端用户活动(从日常任务到复杂的业务逻辑相关活动).应用程序日志记录所有这些任务,并且能够全面记录机构内发生的情况.如果要了解应用程序使用情况.性能或安全性,那么日志管理是最佳 ...

  3. EventLog Analyzer应用程序日志管理

    应用程序日志管理的重要性 应用程序负责组织中的大部分终端用户活动,从简单的日常任务到复杂的业务操作.简而言之,应用程序运行业务,关键业务应用程序,如文件服务器.数据库服务器和邮件服务器,需要不断监测, ...

  4. java程序日志打印规范

    java日志打印规范 一.日志 API 二.日志输出 三.日志配置 V1.0.0_NEW 四.日志性能 五.栈信息打印 一.日志 API 1.[强制]各应用中不可直接使用日志系统(Log4j.Logb ...

  5. java环境陪孩子_Java启蒙之路-Java开发环境搭建与第一个Java程序

    Java开发环境搭建 "工欲善其事,必先利其器",从前面的章节我们了解到,Java程序的运行过程首先编写Java程序,然后再对Java源文件编译,借助JVM运行程序.现在问题来了, ...

  6. Java程序员须知的七个日志管理工具

    本文由 ImportNew - 赖 信涛 翻译自 takipiblog.欢迎加入翻译小组.转载请见文末要求. Splunk vs. Sumo Logic vs. LogStash vs. GrayLo ...

  7. java日志——修改日志管理器配置+日志本地化

    [0]README 0.1) 本文描述+源代码均 转自 core java volume 1, 旨在理解 java日志--修改日志管理器配置+日志本地化 的相关知识: [1]修改日志管理器配置 1.1 ...

  8. java kettle 日志 log_kettle使用log4j管理输出日志

    标签: 在使用kettle进行数据分析和清洗时日志非常多而且杂乱,使用原有的日志有时找不到异常的位置,有时日志不够详细,说简单一点就是日志不是我们想要的.因而对kettle日志进行相应的管理就想得尤为 ...

  9. Java日志管理最佳实践

    原文出处:http://www.ibm.com/developerworks/cn/java/j-lo-practicelog/. 感谢原作者,感谢ibm网站,里面有好多的精华帖. 日志记录是应用程序 ...

最新文章

  1. python网站开发实例-【9】Python接口开发:flask Demo实例
  2. 那些年,我们踩过的 Java 坑
  3. oc基础-self关键字的使用
  4. 成员变量和类变量的区别:
  5. Kinect2.0-空间长度测量
  6. 怎么利用计算机公式计算完成比例,excel表格数据计算所占比例公式的使用教程...
  7. 【转】Java 5种字符串拼接方式性能比较。
  8. web gooflow流程图实现带公式的流程配置
  9. Win7自带驱动备份功能使用教程
  10. 多颗微粒的阵列光镊系统设计
  11. Mac sublime3 在localhsot 上打开项目文件
  12. Java语言Switch语句详解(一)
  13. PCL法线计算及原理
  14. 谁锁了我的帐号?(AD账号的锁定状态查询)
  15. 数组双指针和数组窗口
  16. redis也可以根据经纬度查询附近的元素以及计算两个经纬度的距离???
  17. C++:实现量化GSR模型测试实例
  18. CodeGear 6月8日西安新品发布会 笔记
  19. vmware安装centos7登录出现something went gone 错误
  20. 微信网页授权之回调域名

热门文章

  1. Nancy之区域和分部视图的使用
  2. Codeforces Round #806 (Div. 4)题解
  3. 测人品,看是否满足条件
  4. windows系统用微软账户登录情况下,Hyper-V下虚拟机共享
  5. Skype for Business 网页安排会议(无Exchange Server)
  6. java字节码文件学习
  7. DNS被污染后怎么才能解决?
  8. YOLOv5报错AssertionError:Label class 1 exceeds nc=1 in yolo/dataset.ymal Possible class labels are 0-0
  9. 携创教育:自考本科有用吗?有什么优势?
  10. 超宽带室内信道模型研究与matlab仿真,复杂室内环境超宽带信号信道模型及仿真结果分析.pdf...