我需要破解一个小工具。它应该读取几个文件并转换它们。现在这在我的 IDE 中有效。对于用户,我想添加一个简单的显示日志输出的小 UI。
你知道用于 logback 的即用型 Swing appender 吗?或者将 System.out 重定向到一个只有一个文本字段和一个“关闭”按钮的小 UI 的东西?
PS:我不是在寻找电锯或拼图或莉莉丝。请在应用程序中显示日志消息。
我需要破解一个小工具。它应该读取几个文件并转换它们。现在这在我的 IDE 中有效。对于用户,我想添加一个简单的显示日志输出的小 UI。
你知道用于 logback 的即用型 Swing appender 吗?或者将 System.out 重定向到一个只有一个文本字段和一个“关闭”按钮的小 UI 的东西?
PS:我不是在寻找电锯或拼图或莉莉丝。请在应用程序中显示日志消息。
您需要编写一个自定义附加程序类,如下所示:
public class MyConsoleAppender extends AppenderBase<ILoggingEvent> {
private Encoder<ILoggingEvent> encoder = new EchoEncoder<ILoggingEvent>();
private ByteArrayOutputStream out = new ByteArrayOutputStream();
public MyConsoleAppender() {
LoggerContext lc = (LoggerContext) LoggerFactory.getILoggerFactory();
setContext(lc);
start();
lc.getLogger("ROOT").addAppender(this);
}
@Override
public void start() {
try {
encoder.init(out);
} catch (IOException e) {}
super.start();
}
@Override
public void append(ILoggingEvent event) {
try {
encoder.doEncode(event);
out.flush();
String line = out.toString(); // TODO: append _line_ to your JTextPane
out.reset();
} catch (IOException e) {}
}
}
您可以将 EchoEncoder 替换为 PatternLayoutEncoder(请参阅 logback 示例文件夹中的 CountingConsoleAppender 示例)。
编码器会将每个事件写入一个字节缓冲区,然后您可以提取一个字符串并将其写入您的 JTextPane 或 JTextArea 或您想要的任何内容。
我经常依赖JTextArea#append()
,正如这个例子中所建议的那样。与大多数 Swing 不同,该方法恰好是线程安全的。
没有保修,但这是我刚刚写的一个示例:
/**
* A Logback appender that appends messages to a {@link JTextArea}.
* @author David Tombs
*/
public class JTextAreaAppender extends AppenderBase<ILoggingEvent>
{
private final JTextArea fTextArea;
private final PatternLayout fPatternLayout;
public JTextAreaAppender(final Context loggerContext, final JTextArea textArea)
{
fTextArea = textArea;
// Log the date, level, class name (no package), and the message.
fPatternLayout = new PatternLayout();
fPatternLayout.setPattern("%d{HH:mm:ss.SSS} %-5level - %msg");
fPatternLayout.setContext(loggerContext);
fPatternLayout.start();
// Make sure not to call any subclass methods right now.
super.setContext(loggerContext);
}
@Override
protected void append(final ILoggingEvent eventObject)
{
// Actual appending must be done from the EDT.
SwingUtilities.invokeLater(new Runnable() {
@Override
public void run()
{
final String logStr = fPatternLayout.doLayout(eventObject);
// If the text area already has lines in it, append a newline first.
if (fTextArea.getDocument().getLength() > 0)
{
fTextArea.append("\n" + logStr);
}
else
{
fTextArea.setText(logStr);
}
}
});
}
}