我的应用程序使用System.setErr()
将stderr
更改为最终将关闭的流。在它关闭后,我立即再次使用System.setErr()
将stderr
更改为工作流,这在我的代码执行类似System.err.println()
的操作时很好,但使用java.util.logging
登录控制台(即stderr
)不再有效。
为了解决这个问题,我不得不重新读取日志配置文件,然后自己放回控制台日志处理程序中:
import java.util.logging.ConsoleHandler;
import java.util.logging.Loggger;
import java.util.logging.LogManager;
...
System.setErr(otherStream);
try {
LogManager.getLogManager().readConfiguration();
}
catch(java.io.IOException e) {
// ...
}
Logger.getGlobal().addHandler(new ConsoleHandler());
ConsoleHandler在构造过程中捕获对System.err的引用。另一个解决方案是替换日志树中的所有ConsoleHandler。
LogManager m = LogManager.getLogManager();
synchronized (m) {
Enumeration<String> names = m.getLoggerNames();
while (names.hasMoreElements()) {
Logger logger = m.getLogger(names.nextElement());
if (logger != null) {
for (Handler h : logger.getHandlers()) {
if (h.getClass() == ConsoleHandler.class) {
logger.removeHandler(h);
h.close();
logger.addHandler(new ConsoleHandler());
}
}
}
}
}
System.err.println("This is written to stderr");
// this is stderr
PrintStream oldErr = System.err;
try {
System.setErr(new PrintStream(new File("/path/to/file")));
} catch(Exception e) {
}
System.err.println("This is written to the file");
// note that System.err wouldn't work here because System.err is no longer stderr
// but you preserved stderr in oldErr
try {
System.setErr(oldErr);
} catch(Exception e) {
}
System.err.println("This is written to stderr again");