参考网上博客。
能够输出debug, info, warn, error 四种日志信息。
1. log4j.properties内容如下:
log4j.rootLogger = DEBUG,A1,debug,info,warn,error
#OFF, A1,debug,info,warn,error
# A1 is set to be a ConsoleAppender which outputs to System.out.
log4j.appender.A1=org.apache.log4j.ConsoleAppender
log4j.appender.A1.Threshold = OFF
log4j.appender.A1.layout=org.apache.log4j.PatternLayout
log4j.appender.A1.layout.ConversionPattern=%-r %-d{yyyy\u5E74MM\u6708dd\u65E5 HH\:mm\:ss} %-5p [%t]%x - %m%n
# In this example, we are not really interested in INNER loop or SWAP
# messages. See the effects of uncommenting and changing the levels of
# the following loggers.
log4j.appender.debug=log4j.LogAppender
log4j.appender.debug.Threshold = debug
log4j.appender.debug.DatePattern='_'yyyy-MM-dd
log4j.appender.debug.File=D\:\\log\\debuglog
log4j.appender.debug.layout=org.apache.log4j.PatternLayout
log4j.appender.debug.layout.ConversionPattern=%-r %-d{yyyy\u5E74MM\u6708dd\u65E5 HH\:mm\:ss} %-5p [%t]%x - %m%n
log4j.appender.info=log4j.LogAppender
log4j.appender.info.Threshold = info
log4j.appender.info.DatePattern='_'yyyy-MM-dd
log4j.appender.info.File=D\:\\log\\infolog
log4j.appender.info.layout=org.apache.log4j.PatternLayout
log4j.appender.info.layout.ConversionPattern=%-r %-d{yyyy\u5E74MM\u6708dd\u65E5 HH\:mm\:ss} %-5p [%t]%x - %m%n
log4j.appender.warn=log4j.LogAppender
log4j.appender.warn.Threshold = warn
log4j.appender.warn.DatePattern='_'yyyy-MM-dd
log4j.appender.warn.File=D\:\\log\\warnlog
log4j.appender.warn.layout=org.apache.log4j.PatternLayout
log4j.appender.warn.layout.ConversionPattern=%-r %-d{yyyy\u5E74MM\u6708dd\u65E5 HH\:mm\:ss} %-5p [%t]%x - %m%n
log4j.appender.error=log4j.LogAppender
log4j.appender.error.Threshold = error
log4j.appender.error.DatePattern='_'yyyy-MM-dd
log4j.appender.error.File=D\:\\log\\errorlog
log4j.appender.error.layout=org.apache.log4j.PatternLayout
log4j.appender.error.layout.ConversionPattern=%-r %-d{yyyy\u5E74MM\u6708dd\u65E5 HH\:mm\:ss} %-5p [%t]%x - %m%n
2. log4j 封装包, 包含两个类:LogAppender 和 Logs
LogAppender类代码:
package log4j;
import org.apache.log4j.DailyRollingFileAppender;
import org.apache.log4j.Priority;
/**
* 定义自己的Appender类,继承DailyRollingFileAppender,改写针对Threshold 的设置说明
*@author zhenwei.gu
*@time 2013年7月22日22:10:48
*/
public class LogAppender extends DailyRollingFileAppender {
@Override
public boolean isAsSevereAsThreshold(Priority priority) {
//只判断是否相等,而不判断优先级
return this.getThreshold().equals(priority);
}
}
Logs类代码:package log4j;
/**
* 日志输出代理类。
* <p>
* 主要完成业务代码和日志工具间的解耦,使切换日志工具更加方便。
* <p>
* 如:将Log4j改为common-logging或自己的独立实现,则只需调整该类即可。
* <p>
* 简化了调用方式:如使用log4j,通常用法是先生成一个实例,再调用输出方法。 现在,只需直接使用 Logs.debug();等静态方法即可
*
* @version 1.0
* @since 1.5
*/
public class Logs {
/**
* 是否打印日志,true表示打印日志,false表示不打印。
* <p>
* 开发阶段可以将其设为ture,运行阶段可以设为false
*/
private static final boolean enabled = true;
/** 是否进行源代码定位,ture表示输出源代码所在类以及代码行 */
private static boolean showLocSrc = true;
/** 指定的日志级别 */
private static int level = 1;
/** 日志级别:普通 */
private static final int debug = 1;
/** 日志级别:调试 */
private static final int info = 2;
/** 日志级别:警告 */
private static final int warn = 3;
/** 日志级别:错误 */
private static final int error = 4;
/** 消息所属和消息内容的分隔符 */
private static final String msgSplit = ":";
/** 该类的名称,用于识别该类的堆栈 */
private static final String thisClassName = Logs.class.getName();
/** 默认输出日志的日志工具:log4j */
private static org.apache.log4j.Logger logger = null;
public Logs() {
}
static {
// 可以单独指定log4j的配置文件,也可以使用默认。
// org.apache.log4j.PropertyConfigurator.configure("conf/log4j.properties");
// 得到logger实例,作为输出工具。
// 此句作用是用一个输出实例,取代每个类里面的: Logger.getLogger(X.class)
logger = org.apache.log4j.Logger.getLogger("");
}
/**
* 根据日志级别,输出日志。
* <p>
* 如果要改变日志输出工具,
* <p>
* 如:由原来的log4j改为System.out.println()或logging,则只需改动此类即可。
*
* @param level
* 日志级别
* @param message
* 日志消息
* @param ste
* 堆栈信息。
* <p>
* 如果不需要输出源代码信息,则只需将静态成员属性 showLocSrc设为false即可。
*/
private static void log(int level, Object message, StackTraceElement[] ste) {
if (ste != null) {
// 加入源代码定位
message = getStackMsg(ste) + msgSplit + message;
}
// 转入具体实现,此处为log4j,可以改为其他不同的日志实现。
switch (level) {
case info:
logger.info(message);
break;
case debug:
logger.debug(message);
break;
case warn:
logger.warn(message);
break;
case error:
logger.error(message);
break;
default:
logger.debug(message);
}
}
/**
* 根据堆栈信息得到源代码行信息
* <p>
* 原理:本工具类的堆栈下一行即为源代码的最原始堆栈。
*
* @param ste
* 堆栈信息
* @return 调用输出日志的代码所在的类.方法.代码行的相关信息
* <p>
* 如:com.MyClass 类里的 fun()方法调用了Logs.debug("test");
* <p>
* 则堆栈信息为: com.MyClass.fun(MyClass.java 代码行号)
*/
private static String getStackMsg(StackTraceElement[] ste) {
if (ste == null)
return null;
boolean srcFlag = false;
for (int i = 0; i < ste.length; i++) {
StackTraceElement s = ste[i];
// 如果上一行堆栈代码是本类的堆栈,则该行代码则为源代码的最原始堆栈。
if (srcFlag) {
return s == null ? "" : s.toString();
}
// 定位本类的堆栈
if (thisClassName.equals(s.getClassName())) {
srcFlag = true;
}
}
return null;
}
/**
* 输出info信息
*
* @param message
*/
public static void info(Object message) {
// 如果禁止日志或者输出级别不符,则返回。
if (!enabled || info < level)
return;
if (showLocSrc) {
log(info, message, Thread.currentThread().getStackTrace());
} else {
log(info, message, null);
}
}
/**
* 输出debug信息
*
* @param message
*/
public static void debug(Object message) {
// 如果禁止日志或者输出级别不符,则返回。
if (!enabled || debug < level)
return;
if (showLocSrc) {
log(debug, message, Thread.currentThread().getStackTrace());
} else {
log(debug, message, null);
}
}
/**
* 输出warn信息
*
* @param message
*/
public static void warn(Object message) {
// 如果禁止日志或者输出级别不符,则返回。
if (!enabled || warn < level)
return;
if (showLocSrc) {
log(warn, message, Thread.currentThread().getStackTrace());
} else {
log(warn, message, null);
}
}
/**
* 输出error信息
*
* @param message
*/
public static void error(Object message) {
// 如果禁止日志或者输出级别不符,则返回。
if (!enabled || error < level)
return;
if (showLocSrc) {
log(error, message, Thread.currentThread().getStackTrace());
} else {
log(error, message, null);
}
}
}
3. 使用实例:
log4j.Logs.info(sth);
log4j.Logs.debug(sth);
log4j.Logs.error(sth);
log4j.Logs.warn(sth);