Java日志设计&实践(3) - 开发篇


1.选择恰当的日志级别
2.输出明确的提示文字和充分的现场信息
3.输出内容一行搞定,不要换行
4.其他

 

1.选择恰当的日志级别



选择日志级别时需要遵循一些通用规范,不可随意定义

log4j的日志级别,由低到高排列:all trace debug info warn error fatal off
其中,all off仅用以log4j配置文件中开启或关闭所有日志,trace fatal一般也用不到
对于开发人员来说,只需要关注debug info warn error

debug
    正常情况下不需要输出,只有当出问题时才需要输出的日志信息,由于生产环境无法单步调试,可以把debug级别的日志想象成你在生成环境中debug
info
    可能要关注或者只有比较重要的信息才需要输出,如:用户登录、退出、后台job执行时长等
warn
    存在一些潜在的危险时输出的日志,比如:请求参数中包含攻击注入脚本
error
    如:请求数据库时的SQL异常

当然,最好的方法还是参考优秀的开源代码
   

2.输出明确的提示文字和充分的现场信息


要点:
1)明确的提示文字,看到这段提示文本就可以知道发生了什么,不需要再去扒拉源码
2)充分的现场信息,如:用户信息、引发异常的参数值、异常栈信息等

举例:
LOG.warn("Unknown value for includeParams parameter to URL tag: " + includeParams);
LOG.warn("Unable to put request parameters (" + extractQueryString() + ") into parameter map.", e);
LOG.warn("Could not find token mapped to token name " + tokenName);

 

3.一条日志一行搞定


这个是为了方便跟踪和分析日志,使用grep命令时不至于仅看到一条日志的部分内容

 

4.其他

 

4.1.尽量使用一套日志接口,强烈推荐slf4j


两大理由:
1)使用{}占位符,避免字符串拼接

以刚才三个log为例
LOG.warn("Unknown value for includeParams parameter to URL tag: " + includeParams);
LOG.warn("Unable to put request parameters (" + extractQueryString() + ") into parameter map.", e);
LOG.warn("Could not find token mapped to token name " + tokenName);

如果改用slf4j的话,写法如下:
log.warn("Unknown value for includeParams parameter to URL tag: {}", includeParams);
log.warn("Unable to put request parameters ({}) into parameter map.", extractQueryString(), e);
log.warn("Could not find token mapped to token name {}", tokenName);


2)执行实际日志输出前强制检查log是否开启

组合使用log4j+slf4j时,执行的warn方法实际是这样的:
  public void warn(String format, Object arg) {
    if (logger.isEnabledFor(Level.WARN)) {
      FormattingTuple ft = MessageFormatter.format(format, arg);
      logger.log(FQCN, Level.WARN, ft.getMessage(), ft.getThrowable());
    }
  }

 

4.2.不要使用System.out.println()


这个就不多说了,日志中看到一句莫名其妙hello,world你会怎么想,怎么查????

4.3.不要使用e.printStackTrace()


这种打印只能输出到catalina.out中,无法单独制定输出目的文件,还会导致日志输出混乱

4.4.slf4j打印异常堆栈信息


两个例子:

try { 
    if (true) { 
        throw new RuntimeException("i'm ok"); 
    } 
} catch (Exception e) { 
    log.error("Error. param:{}, param2:{}, param3:{}", param, param2, param3, e); 
}

将打印:
ERROR 2015-01-17 15:11:51,426 Error. param:0, param2:2, param3:false [cn.xxt.log.test.Slf4jTest.main(Slf4jTest.java:36)]
java.lang.RuntimeException: i'm ok
    at cn.xxt.log.test.Slf4jTest.main(Slf4jTest.java:33) 
   
try { 
    if (true) { 
        throw new RuntimeException("i'm sorry"); 
    } 
} catch (Exception e) { 
    log.error("Error. param:{}, param2:{}, param3:{}, {}", param, param2, param3, e); 
}

将打印:
ERROR 2015-01-17 15:11:51,429 Error. param:0, param2:2, param3:false, java.lang.RuntimeException: i'm sorry [cn.xxt.log.test.Slf4jTest.main(Slf4jTest.java:44)]

差异:前者输出了异常栈信息,后者没有
原因:后者用{}占位符打印异常对象e,导致异常栈信息没有输出

 

参考文档


为什么要使用SLF4J而不是Log4J
http://www.importnew.com/7450.html

posted on 2015-01-17 15:54  xxt-mov  阅读(684)  评论(0编辑  收藏  举报

导航