其实这不是什么神奇的trick,只是很多人包括我在一直以来使用Java的日志框架比如log4j时没有意识到可以动态改变日志级别。如果你看到别人这么写了的话,很快就会转变观念的,我也是。
https://github.com/xnnyygn/dynamic-log-level 这个github是我整理的log4j动态改变日志级别的代码以及测试,有兴趣的可以参考下。
个人觉得这个trick可以用的地方还是程序员给自己开“后门”,特别是查线上问题时。由于很多人打印日志的级别的convention不一致。比如我喜欢非常详细的输入参数和输出结果用DEBUG,但是有些人就喜欢打INFO。一些复杂的逻辑有些人打了十几八行的INFO级别日志,看得我心烦,这帮人打INFO就像喝水一样……所以有些时候临时改变日志级别可能是有用的。不过从开发上来说,没有很好地统一日志级别规范,没有重视CR,是导致这个问题的主因,但是作为程序员要给自己留一手,比如动态改变日志级别之类的。
言归正题,具体代码其实不复杂,以下展示下测试代码和实现代码。
测试代码如下
import org.apache.log4j.Logger; import org.junit.Test; public class Log4jLogLevelManagerTest { private static final Logger logger = Logger.getLogger("test"); @Test public void test() { Log4jLogLevelManager manager = new Log4jLogLevelManager(); // original level of logger 'test' is INFO logger.info("info message should be outputed"); logger.debug("debug message should not be outputed"); // change log level to debug manager.changeLogLevel("test", "DEBUG"); logger.debug("debug message should be outputed now"); // reset level of logger 'test' manager.resetLogLevel("test"); logger.debug("debug message should not be outputed again"); } }
运行时会产生类似如下信息
2015-05-16 08:10:16,295 INFO [main] dynamicloglevel.Log4jLogLevelManagerTest (Log4jLogLevelManagerTest.java:23) - info message should be outputed 2015-05-16 08:10:16,301 DEBUG [main] dynamicloglevel.Log4jLogLevelManagerTest (Log4jLogLevelManagerTest.java:29) - debug message should be outputed now
实现代码如下
import java.util.concurrent.ConcurrentHashMap; import java.util.concurrent.ConcurrentMap; import org.apache.log4j.Level; import org.apache.log4j.LogManager; import org.apache.log4j.Logger; public class Log4jLogLevelManager implements LogLevelManager { private ConcurrentMap<String, Level> levels = new ConcurrentHashMap<String, Level>(); public void changeLogLevel(String loggerName, String level) { Logger logger = determineLogger(loggerName); if (logger == null) return; // logger not found // push original level in the first time of changing levels.putIfAbsent(loggerName, logger.getLevel()); logger.setLevel(Level.toLevel(level)); } private Logger determineLogger(String loggerName) { if ("ROOT".equals(loggerName)) return LogManager.getRootLogger(); // don't use LogManager#getLogger here since getLogger will cause // making of new logger if logger not found return LogManager.exists(loggerName); } public void resetLogLevel(String loggerName) { Logger logger = determineLogger(loggerName); if (logger == null) return; // logger not found Level originalLevel = levels.get(loggerName); if(originalLevel == null) return; // level of logger is not changed logger.setLevel(originalLevel); } }
注意这里使用了ConcurrentMap,考虑到类似web环境下并发的可能,这里还是加上并发控制比较好。注意使用原子操作,比如putIfAbsent。
总体来说,这个trick不是很复杂,就当是笔记好了,说不定哪天可以使用。