这几天在为自己写的一个网络爬虫加一个UI控制界面,之前的爬虫核心是以命令行启动的,所以日志信息由log4j直接输出的控制台,可是现在有了UI,就不能再将日志信息输出到控制台了,必须将日志信息以某种方式截取,输出到界面上。
在网上找了一下相关代码,确实不少,但经过实践检验之后,发现要么是代码太多太麻烦,要么是性能太差(爬虫运行时每秒钟产生日志信息超过百行很随意),大都不适用。无奈,只好自己绞尽脑汁,才勉强做出来一个能用的。现在把代码贴出来,供学习交流,欢迎拍砖!
废话不多说,看代码:
/* * To change this template, choose Tools | Templates * and open the template in the editor. */ package cn.ysh.studio.gui.log; import java.io.IOException; import java.io.PipedReader; import java.io.PipedWriter; import java.io.Writer; import org.apache.log4j.Logger; import org.apache.log4j.Appender; import org.apache.log4j.WriterAppender; /** * * 类描述: * 重置log4j的Appender的Writer * @author 杨胜寒 * @date 2011-12-20 创建 * @version 1.0 */ public abstract class LogAppender extends Thread { protected PipedReader reader; public LogAppender(String appenderName) throws IOException { Logger root = Logger.getRootLogger(); // 获取子记录器的输出源 Appender appender = root.getAppender(appenderName); // 定义一个未连接的输入流管道 reader = new PipedReader(); // 定义一个已连接的输出流管理,并连接到reader Writer writer = new PipedWriter(reader); // 设置 appender 输出流 ((WriterAppender) appender).setWriter(writer); } }
这个类是一个基类,实际上是不能够直接使用的,由它的子类负责将来自控制台的日志信息输出到UI组件。
比如现在需要将日志信息截获,输出到一个JLabel组件,代码如下:
/* * To change this template, choose Tools | Templates * and open the template in the editor. */ package cn.ysh.studio.gui.log; import java.io.IOException; import java.util.Scanner; import javax.swing.JLabel; /** * * 类描述: * 不间断地扫描输入流 * 将扫描到的字符流显示在JLabel上 * @author 杨胜寒 * @date 2011-12-20 创建 * @version 1.0 */ public class LabelLogAppender extends LogAppender { private JLabel label; /** * 默认的构造 * @param label 记录器名称,该记录器输出的日志信息将被截取并输出到指定的JLabel组件 * @throws IOException */ public LabelLogAppender(JLabel label) throws IOException { super("label"); this.label = label; } @Override public void run() { // 不间断地扫描输入流 Scanner scanner = new Scanner(reader); // 将扫描到的字符流显示在指定的JLabel上 while (scanner.hasNextLine()) { try { //睡眠 Thread.sleep(100); String line = scanner.nextLine(); label.setText(line); line = null; } catch (Exception ex) { //异常信息不作处理 } } } }
实际上,将日志信息输出到JTextArea或其他多行文本组件更加常见,比如MyEclipse或NetBeans等IDE。那么一下面的代码就展示了如何将日志截获并输出到JTextArea组件,同时自动使垂直滚动条跟随。请看到代码:
/* * To change this template, choose Tools | Templates * and open the template in the editor. */ package cn.ysh.studio.gui.log; import java.io.IOException; import java.util.Scanner; import javax.swing.JScrollPane; import javax.swing.JTextArea; /** * * 类描述: * 不间断地扫描输入流 * 将扫描到的字符流显示在JTextArea上 * @author 杨胜寒 * @date 2011-12-20 创建 * @version 1.0 */ public class TextAreaLogAppender extends LogAppender { private JTextArea textArea; private JScrollPane scroll; /** * 默认的构造 * @param textArea 记录器名称,该记录器输出的日志信息将被截取并输出到指定的JTextArea组件 * @param scroll JTextArea组件使用的滚动面板,因为在JTextArea中输出日志时,默认会使垂直滚动条自动向下滚动,若不需要此功能,此参数可省略 * @throws IOException */ public TextAreaLogAppender(JTextArea textArea, JScrollPane scroll) throws IOException { super("textArea"); this.textArea = textArea; this.scroll = scroll; } @Override public void run() { // 不间断地扫描输入流 Scanner scanner = new Scanner(reader); // 将扫描到的字符流输出到指定的JTextArea组件 while (scanner.hasNextLine()) { try { //睡眠 Thread.sleep(100); String line = scanner.nextLine(); textArea.append(line); textArea.append("\n"); line = null; //使垂直滚动条自动向下滚动 scroll.getVerticalScrollBar().setValue(scroll.getVerticalScrollBar().getMaximum()); } catch (Exception ex) { //异常不做处理 } } } }
按照上述方式封转之后,他们的使用就非常简单了。在窗体组件绘制完成后,就可以启动他们了:
/* * To change this template, choose Tools | Templates * and open the template in the editor. */ package cn.ysh.studio.gui.test; import cn.ysh.studio.gui.log.LabelLogAppender; import cn.ysh.studio.gui.log.TextAreaLogAppender; import javax.swing.JFrame; import javax.swing.JLabel; import javax.swing.JOptionPane; import javax.swing.JScrollPane; import javax.swing.JTextArea; import org.apache.commons.logging.Log; import org.apache.commons.logging.LogFactory; /** * * 类描述: * 测试日志输出到UI组件 * @author 杨胜寒 * @date 2011-12-24 创建 * @version 1.0 */ public class LogDemoJFrame extends JFrame { private JLabel logLabel; private JScrollPane logScrollPane; private JTextArea logTextArea; private final static Log log = LogFactory.getLog(LogDemoJFrame.class); public LogDemoJFrame() { logLabel = new javax.swing.JLabel(); logScrollPane = new javax.swing.JScrollPane(); logTextArea = new javax.swing.JTextArea(); setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE); logLabel.setText(" "); logTextArea.setColumns(20); logTextArea.setRows(5); logScrollPane.setViewportView(logTextArea); javax.swing.GroupLayout layout = new javax.swing.GroupLayout(getContentPane()); getContentPane().setLayout(layout); layout.setHorizontalGroup( layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING).addGroup(layout.createSequentialGroup().addContainerGap().addComponent(logLabel, javax.swing.GroupLayout.DEFAULT_SIZE, 610, Short.MAX_VALUE).addContainerGap()).addComponent(logScrollPane, javax.swing.GroupLayout.Alignment.TRAILING, javax.swing.GroupLayout.DEFAULT_SIZE, 630, Short.MAX_VALUE)); layout.setVerticalGroup( layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING).addGroup(layout.createSequentialGroup().addContainerGap().addComponent(logLabel).addPreferredGap(javax.swing.LayoutStyle.ComponentPlacement.RELATED).addComponent(logScrollPane, javax.swing.GroupLayout.DEFAULT_SIZE, 305, Short.MAX_VALUE))); pack(); } public void initLog() { try { Thread t1, t2; t1 = new LabelLogAppender(logLabel); t2 = new TextAreaLogAppender(logTextArea, logScrollPane); t1.start(); t2.start(); } catch (Exception e) { JOptionPane.showMessageDialog(this, e, "绑定日志输出组件错误", JOptionPane.ERROR_MESSAGE); } } public static void main(String[] s) { LogDemoJFrame logDemoFrame = new LogDemoJFrame(); logDemoFrame.initLog(); logDemoFrame.setVisible(true); for (int i = 0; i < 1000; i++) { log.info("测试日志输出:" + i); } } }
很简单的一个Demo,仅供参考。
下面是我的Log4j日志配置信息:
log4j.appender.label=org.apache.log4j.ConsoleAppender log4j.appender.label.layout=org.apache.log4j.PatternLayout log4j.appender.label.layout.ConversionPattern=%m%n log4j.appender.textArea=org.apache.log4j.ConsoleAppender log4j.appender.textArea.layout=org.apache.log4j.PatternLayout log4j.appender.textArea.layout.ConversionPattern=%-d{yyyy-MM-dd HH:mm:ss} [ %p ] %m%n log4j.rootLogger=INFO,label,textArea
其实鉴于上述方法,还可以实现更多自定义log4j日志输出功能,在此不再赘述。
原创文章,转载请注明出处: http://www.yshjava.cn/post/322.html
相关推荐
首先,Log4J是由Apache软件基金会开发的一个开源日志组件,它提供了灵活的日志记录功能,包括不同的日志级别(如DEBUG、INFO、WARN、ERROR等)、自定义日志格式和多种输出方式。在Android上,由于系统环境的限制,...
本文档主要针对初学者,详细介绍了如何使用CANoe软件截取实车CAN通信日志(简称CANoe Log),并对其进行分析。以下为本教材所涉及的详细知识点: 1. CANoe Log的截取方法: - 首先打开CANoe软件。 - 调出Log设置...
在Unity开发过程中,我们能很容易在Console控制台看到日志输出,但往往当测试人员测试时,使用的是pc包测试,想要看日志就不是那么方便,为了方便排错找错,我们希望在运行时的 Game Window 中有一个 Debug 控制台更...
标题“js截取div并输出为图片”所描述的就是这样的一个功能,通过JavaScript实现对指定`div`元素的截图并转化为图片格式。在这个项目中,以截取百度地图为例,解决了SVG元素无法正常截取的问题,这对于包含复杂SVG...
【电脑输出音频截取】是一种技术,用于捕获计算机的声音输出,这在处理网络上的多媒体内容,特别是Flash视频和音乐时非常有用。这种技术允许用户记录来自任何正在播放的声音源,比如在线播放的歌曲、电影对话或者...
本文将详细讲解如何在CentOS7环境下,根据特定的时间段截取Tomcat的日志并保存到指定文件,这对于排查问题、监控系统性能以及进行故障排除都非常有帮助。 首先,我们要了解Tomcat的日志文件。在默认配置下,Tomcat...
4. **日志安全**:注意不要在生产环境中泄露敏感信息,比如用户数据或应用密钥,确保在日志输出前进行适当处理。 总之,日志分析是Android开发中不可或缺的一部分,logview工具的引入使得这一过程更加高效和便捷。...
组件在截取成功后(用户点击保存按钮后),将截取的图片临时路径存放到app.globalData.screenshotimagepath,并进行页面回退1级操作。 在原页面的onshow方法中可以对app.globalData.screenshotimagepath进行判断,...
MATLAB GUI(图形用户界面)制作的播放器是一种利用MATLAB强大的编程能力构建的交互式音频播放工具。MATLAB不仅支持数值计算,还提供了丰富的GUI工具箱,使得开发者可以创建出具有直观界面的应用程序,例如这个...
在这个例子中,`ilog3.h`是包含日志系统接口的头文件,`init_logger`函数初始化日志系统并指定日志文件,`log_message`则用于打印日志信息。 总的来说,理解和实现C语言的日志系统需要对文件操作、字符串处理和条件...
在MATLAB环境中,GUI(图形用户界面)是一种强大的工具,用于构建交互式应用程序,而“MATLAB下的数据截取工具”就是这样的一个应用。这个工具是利用GUIDE(图形用户界面开发环境)创建的,目的是简化对大量数据的...
Delphi7摄像头图像截取组件源代码以及窗体皮肤组件源代码 Delphi7摄像头图像截取组件源代码以及窗体皮肤组件源代码 Delphi7摄像头图像截取组件源代码以及窗体皮肤组件源代码 Delphi7摄像头图像截取组件源代码以及...
主要用于sql数据库日志文件超级大,但是用正常的收缩日志方式不成功。自己已经成功使用。
Java 调用exe程序,并且获取exe程序的输出
利用管道流截取JAVA项目的控制台输出,可以实现多线程写入,并输出控制台信息
开发者需要熟练掌握MATLAB编程,理解GUI组件的工作原理,并能够灵活运用函数和回调机制来响应用户的操作。此外,由于MATLAB并非专门的多媒体处理工具,所以在处理大型或高码率的媒体文件时,性能可能会成为挑战,...
matlab做的简单的图像处理应用。GUI。噪声,滤波,旋转,反色,截取等功能。
PDFApp截取工具是一款基于Python开发的实用工具,主要用于帮助用户高效、灵活地截取PDF文档中的特定页面,并将这些截取的页面保存到用户指定的文件位置。这个压缩包包含以下三个文件: 1. PdfApp.exe:这是主应用...
1、这里截取当前时间的年月日,不同系统可能默认的不一致,自己可通过cmd–...这里由于日志文件命名是有规律的,都是日志名+时间(年月日).log,通过截取当中的日期与当前时间的一个月前日期比较来决定删除对应的文件。