PrintStream类的学习与多目的地输出重定向问题

原创 2012年03月22日 15:19:04

最近开发的Dodo工具箱中需要一个日志功能,当然现在已经有很成熟很好用的Log4j,但是我觉得作为一个菜鸟,有必要自己开发一个简单的日志处理模块。下面简介该模块的设计与输出重定向这个难点。

用例设计:


类设计:


实现难点——输出重定向:

        我是这样设想的,日志来源于三个方面 1其他模块使用Logger记录的日志信息 2 系统执行过程中需要输出的信息 3 系统在运行过程中碰到的错误信息。日志输出的目的也有三个地方,第一:控制台 第二:日志文件 第三:gui即时信息窗口。

        如果只是单目的地的重定向十分简单,比如说想把System.err的信息重定向到文件out,只需在系统启动时调用如下两行即可:

PrintStream out = new PrintStream(new BufferedOutputStream(new FileOutputStream("out")));
System.setErr(out);
           

        现在的难点在于需要将错误信息输出到多个目的地(文件,控制台,gui窗口),没有一个适合的PrintStream的子类来完成这个任务,那么我们就需要继承PrintStream,并在这个子类中重新print方法来完成我们的输出逻辑,我定义的该类的名称为IORedirect,该类的源码如下:

package toolBox.core.utility;

import java.io.FileNotFoundException;
import java.io.PrintStream;
import java.util.Date;
import java.util.Vector;

import com.ibm.icu.text.SimpleDateFormat;

/**
 * The class pipelines print/println's to several PrintStream. Useful for
 * directing system.out and system.err to external files etc.
 * 
 * @author jiangkai
 * 
 */
public class IORedirect extends PrintStream {
	
	protected static SimpleDateFormat dateformat = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
	
	/**The default target output stream*/
	protected PrintStream default_target=null;

	/**The different target output streams*/
	protected Vector<PrintStream> targets=null;
	
	/** */
	protected Vector<Boolean> appendHeader = null;
	
	/** */
	protected String prefix = null;
	
	/**
	 * initializes the object, with a default printstream.
	 */
	public IORedirect(PrintStream s) {
		this(s,"defaultoutput");
	}
	
	public IORedirect(PrintStream s,String p)
	{
		super(s);
		default_target = s;
		prefix = p;
		
		targets = new Vector<PrintStream>();
		appendHeader = new Vector<Boolean>();
		
		clear();
	}
	
	/**
	 * removes all streams and places the default printstream, if any, again in
	 * the list.
	 * 
	 * @see #getDefault()
	 */
	public void clear() {
		
		targets.clear();
		appendHeader.clear();
		
		if (getDefault() != null)
		{
			targets.add(getDefault());
			appendHeader.add(true);
		}
	}

	/**
	 * returns the default printstrean, can be NULL.
	 * 
	 * @return the default printstream
	 * @see #m_Default
	 */
	public PrintStream getDefault() {
		return default_target;
	}

	public void defaultPrint(String x)
	{
		if(default_target!=null)
			default_target.print(x);
	}
	
	public void defaultPrintln(String x)
	{
		if(default_target!=null)
			default_target.println(x);
	}
	
	/**
	 * adds the given PrintStream to the list of streams, with NO timestamp and
	 * NO prefix.
	 * 
	 * @param p
	 *            the printstream to add
	 */
	public void addPrintStream(PrintStream p) {
		addPrintStream(p,false);
	}

	public void addPrintStream(PrintStream p,boolean ah)
	{
		if (!targets.contains(p))
		{
			targets.add(p);
			appendHeader.add(ah);
		}
	}
	
	public void removePrintStream(PrintStream p) {
		if (targets.contains(p))
		{
			int index = targets.indexOf(p);
			targets.remove(index);
			appendHeader.remove(index);
		}
	}

	public boolean containsPrintStream(PrintStream p) {
		return targets.contains(p);
	}

	public int size() {
		return targets.size();
	}

	public void flush() {
		for (PrintStream element : targets)
			element.flush();
	}

	protected void printHeader()
	{
		for(PrintStream element:targets)
			if(appendHeader.get(targets.indexOf(element)))
				element.print(dateformat.format(new Date())+" "+prefix+"\n\t");
	}
	
	@Override
	public void print(int x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(long x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(float x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(double x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(boolean x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(char x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(char[] x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(String x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void print(Object x) {
		printHeader();
		for (PrintStream element : targets)
			element.print(x);
		flush();
	}

	@Override
	public void println(int x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(long x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(float x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(double x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(boolean x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(char x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(char[] x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(String x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	@Override
	public void println(Object x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}

	/**
	 * Writes <code>len</code> bytes from the specified byte array starting at
	 * offset <code>off</code> to this stream. If automatic flushing is enabled
	 * then the <code>flush</code> method will be invoked.
	 * 
	 * <p>
	 * Note that the bytes will be written as given; to write characters that
	 * will be translated according to the platform's default character
	 * encoding, use the <code>print(char)</code> or <code>println(char)</code>
	 * methods.
	 * 
	 * @param buf
	 *            A byte array
	 * @param off
	 *            Offset from which to start taking bytes
	 * @param len
	 *            Number of bytes to write
	 */
	public void write(byte buf[], int off, int len) {
		for (PrintStream element : targets)
			element.write(buf, off, len);
		flush();
	}

	/**
	 * Writes the specified byte to this stream. If the byte is a newline and
	 * automatic flushing is enabled then the <code>flush</code> method will be
	 * invoked.
	 * 
	 * <p>
	 * Note that the byte is written as given; to write a character that will be
	 * translated according to the platform's default character encoding, use
	 * the <code>print(char)</code> or <code>println(char)</code> methods.
	 * 
	 * @param b
	 *            The byte to be written
	 * @see #print(char)
	 * @see #println(char)
	 */
	public void write(int b) {
		for (PrintStream element : targets)
			element.write(b);
		flush();
	}

	public static void main(String[] args) throws FileNotFoundException
	{
		IORedirect stdRedirect = new IORedirect(System.err,"stderr");
//		stdRedirect.addPrintStream(System.err);
		System.setErr(stdRedirect);
		System.err.println("error");
		//stdRedirect.addPrintStream(System.out);
//		System.setOut(stdRedirect);
//		System.out.println("no");
	}
}
              现在一目了然了,我们的输出逻辑在于重写PrintStream类中的Print函数,在该函数中我们自己定义需要向那写输出流输出日志信息,我在这里的写法是
	@Override
	public void println(boolean x) {
		printHeader();
		for (PrintStream element : targets)
			element.println(x);
		flush();
	}
对于targets中的每一个输出流输出信息,当然,你可以在这边直接写文件输出的逻辑,窗口输出的逻辑,但是当前这种写法使得IORedirect这个类更加通用,不管你是什么输出流,只要注册到该IORediect类即可。

          下面贴出该类的使用办法:

		IORedirect stdoutRedirect = new IORedirect(System.out, "stdout");
		System.setOut(stdoutRedirect);
		stdoutRedirect.addPrintStream(m_windowStream, true);
		stdoutRedirect.addPrintStream(m_FileStream,true);


这样,即将m_windowStream和m_FileStream注册到了该重定向对象中,这样系统的输出会输出到m_windowStream和m_FileStream这两个输出流中去,当然了,m_windowStream和m_FileStream也必须是PrintStream的子类,关于写法很简单,这里不再啰嗦。


相关文章推荐

如何修改int的打印内容——史上最难的JAVA面试题

要求方法1在被调用之后打印出a=100 b=200 请写出method1代码

Apache Rewrite Module 的重定向问题

  • 2010年05月18日 20:19
  • 1KB
  • 下载

Windows 命令行输入输出重定向问题

最近学校的网比较搓,DNS天天挂,出口带宽天天堵,NAT后的总出口带宽也才4MB/s(来源:360测速),唉,不亲身体会鬼才知道一堆人共享这个带宽是什么感觉。 废话不多说了,在Unix下重定向用...
  • hxh129
  • hxh129
  • 2013年02月28日 19:38
  • 4970

F5与应用的重定向配合问题

  • 2015年05月07日 10:03
  • 647KB
  • 下载

console的使用及引申的输入输出重定向问题

为了解决Scanner类输入可见的问题,java SE 6 特别引入了Console类来实现这个目的,以下是Console的使用示例: import java.io.Console; impor...

C语言输入输出的重定向问题

1. C语言是没有I/O的语言   C语言本身就是没有I/O能力的语言,是依靠函数模块来完成的。如printf()就是一个I/O函数,在编译时,编译器并不编译printf()函数,而把他留在链接(...

关于学习algs4时用eclipse写程序时重定向的问题

在Algs4的1.1.10中需要用到重定向,来输入两个文件tingW.txt和tinyT.txt文件,但是好不容配置好...
  • shz2050
  • shz2050
  • 2014年05月07日 22:27
  • 1353

servlet---知识点深化---重定向/转发/字节流字符流共存问题/多个servlet输出问题/请求包含

一、重定向 1)无论第一个是doGet还是doPost,第二个走的都是doGet (重定向实际上只是给第一个servlet返回新地址而已,此时第一个servlet也只能通过get方式访问) 2)传参:...
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:PrintStream类的学习与多目的地输出重定向问题
举报原因:
原因补充:

(最多只允许输入30个字)