【问题标题】:java.io.StreamCorruptedException: invalid stream header when writing to the stdout in child while communicating child and parent, ProcessBuilderjava.io.StreamCorruptedException:在子进程和父进程通信时写入子进程中的标准输出时流标头无效,ProcessBuilder
【发布时间】:2023-10-30 22:16:02
【问题描述】:

我有一个代码段,其中创建了一个新的子进程,并且需要将一些新的进程操作结果从子进程发送到父进程。因此,我为子类ObjectOutputStream stream = new ObjectOutputStream(System.out); 的标准输出创建了一个新的 ObjectOutputStream,并将子进程中的对象序列化并发送到父进程,反序列化在父进程中完成。

它运行良好,没有任何问题。但是当我尝试在子代码中使用System.out.println() 时,问题就来了,它也在写入标准输出。父进程也尝试反序列化System.out.println(),然后父进程就会出现异常。

课程详情如下

public class DTO implements Serializable{
    private static final long serialVersionUID = 1L;
    private String name;

    public DTO()
    {
        this.name = "name";
    }

    public String getName() {
        return name;
    }

@Override
    public int hashCode() {}

@Override
public boolean equals(Object obj) {}

父类.java

public class Parent {

  public static void main(String[] args) {

      try {
          new Parent().start();
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

      public void start() throws IOException, InterruptedException, ClassNotFoundException
      {
            String classpath = System.getProperty("java.class.path");
            String className = Child.class.getCanonicalName();

            ProcessBuilder builder = new ProcessBuilder(
                "java", "-cp", classpath, className);

            Process process = builder.start();

            if (process.isAlive()) {

                ObjectInputStream input = new ObjectInputStream(process.getInputStream());
                DTO dto = (DTO)input.readObject();

            }
      }
}

子.java

public class Child {

    public static void main(String[] args) throws IOException {
        DTO dto = new DTO();

        System.out.println("printing random text here");

        ObjectOutputStream stream = new ObjectOutputStream(System.out);
        stream.writeObject(dto);
        stream.flush();
        stream.close();
    }
}

例外

java.io.StreamCorruptedException: invalid stream header: 64617364
    at java.io.ObjectInputStream.readStreamHeader(Unknown Source)
    at java.io.ObjectInputStream.<init>(Unknown Source)
    at working.Parent.start(Parent.java:35)
    at working.Parent.main(Parent.java:14)

请注意,如果我们删除 child.java 中使用的System.out.println(),程序将不会出错地执行。我认为这是因为序列化和sysout 都写入标准输出,所以父级认为它可以对两者进行反序列化。任何解决此问题的建议或其他类型的方法都将不胜感激

【问题讨论】:

    标签: java io processbuilder objectinputstream


    【解决方案1】:

    您通过直接向其写入字符串破坏了流。所以你有一个StreamCorruptedException。那里并不奇怪。你不能那样做。您不能反序列化还包含随机插值System.out.println()s 的对象流。这应该是显而易见的。

    任何解决此问题的建议

    是的。不要这样做。

    或替代方法

    替代方法是什么?

    您只需决定您使用System.out 做什么。 或者它是一个对象输出流或者它用于打印文本。不能同时进行。

    NB 你还没有输入任何代码来读取错误流,正如我告诉你的last time you posted this code。没有错误检查就无法部署代码。

    【讨论】: