System.out的多向重定向

时间:2013-07-16 22:28:25

标签: java stream

使用以下示例中显示的方法,当其他类拦截输出流时,输出流将丢失。如何修复我的程序?

Console.java:

package interception;

import java.awt.BorderLayout;
import java.awt.EventQueue;

import javax.swing.*;
import javax.swing.border.EmptyBorder;

import java.io.*;

public class Console extends JFrame 
{

    private JPanel contentPane;
    private JTextPane textPane;

    private class Interceptor extends PrintStream
    {
        public Interceptor(OutputStream out)
        {
            super(out,true);
        }

        @Override
        public void print(String s)
        {
            super.print(s);
            textPane.setText(textPane.getText()+s);
        }

    }

    public static void main(String[] args) {
        EventQueue.invokeLater(new Runnable() {
            public void run() 
            {
                try 
                {
                    Console frame = new Console();
                    frame.setVisible(true);

                    System.out.print("System standard output stream test\n");
                    new Troll();
                    System.out.print("Haters gonna hate\n");
                } catch (Exception e) {
                    e.printStackTrace();
                }
            }
        });
    }

    public Console() 
    {
        setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
        setBounds(100, 100, 450, 300);
        contentPane = new JPanel();
        contentPane.setBorder(new EmptyBorder(5, 5, 5, 5));
        contentPane.setLayout(new BorderLayout(0, 0));
        setContentPane(contentPane);

        textPane = new JTextPane();
        textPane.setEditable(false);
        contentPane.add(textPane, BorderLayout.CENTER);

        redirectStream();
    }

    protected void redirectStream()
    {
        PrintStream iout = new Interceptor(System.out);
        System.setOut(iout);        
    }
}

Troll.java:

package interception;

import java.io.*;

public class Troll 
{
    private class Interceptor extends PrintStream
    {
        public Interceptor(OutputStream out)
        {
            super(out,true);
        }

        @Override
        public void print(String s)
        {
            super.print(s);
        }
    }

    public Troll()
    {
        PrintStream iout = new Interceptor(System.out); 
        System.setOut(iout);

        System.out.print("I've stolen your stream bwahaha!\n");
    }
}

enter image description here

换句话说 - 在构建“Troll”之后,控制台不再能够拦截输出流,因此文本“讨厌讨厌”并未在其中显示。这就是问题所在。

更新

Troll类的关闭流显然是不可接受的解决方案。

1 个答案:

答案 0 :(得分:0)

您认为Troll.Interceptor.print()会自动调用Console.Interceptor中的print()方法吗?这不会发生。当Troll.Interceptor.print()调用super.print(s)时,它调用为PrintStream定义的print()方法。您已使用 OutputStream 构建PrintStream,而OutputStream没有print()方法。 PrintStream逻辑可能会在底层的OutputStream上调用write()(并且write()没有被覆盖),但是没有办法调用print(),因为PrintStream.print()没有看到“out”另一个PrintStream。

我不知道这对你有多大帮助,因为我不清楚你期望它做什么。