读取从C#客户端发送的ObjectInputStream时出现“StreamCorruptedException:invalid stream header”

时间:2012-11-01 23:04:07

标签: c# java stream

问题

我正在尝试从C#客户端向此Java Server发送一个protobuf消息,但我得到了这个例外:

java.io.StreamCorruptedException: invalid stream header: 0A290A08 
java.io.StreamCorruptedException: invalid stream header: 0A290A08
    at java.io.ObjectInputStream.readStreamHeader(Unknown Source)
    at java.io.ObjectInputStream.<init>(Unknown Source)

说实话,我有点不知所措。任何帮助表示赞赏。谢谢!

  • Java服务器
    public ControllerThread(Socket s){
    this.s = s; try {

        this.objectInputStream = new ObjectInputStream(s.getInputStream());
        byte size = objectInputStream.readByte();System.out.println("Server: BYTES SIZE:" +     size);
        byte[] bytes = new byte[size];
        objectInputStream.readFully(bytes);
        AddressBook adb = AddressBook.parseFrom(bytes);
        System.out.println("Server: Addressbook:" + adb.getPersonCount());

    } catch (IOException e) { 
        System.out.println("Server: BufferedReader oder PrintWriter von ThermoClient konnte nicht erstellt werden");
        e.printStackTrace(); } 
        } }

C#代码

public AddressBook InitializeAdressBook()
{
    Person newContact = new Person();

    AddressBook addressBookBuilder = new AddressBook();
    Person john = new Person();
    //john.id=1234;
    john.name="John Doe";
    john.email="jdoe@example.com";
    Person.PhoneNumber nr = new Person.PhoneNumber();
    nr.number="5554321";
    john.phone.Add(nr);
    addressBookBuilder.person.Add(john);
    TextBox.Text += ("Client: Initialisiert? " + addressBookBuilder.ToString()) + "\t" + "\n";
    TextBox.Text += " Erster Person " + addressBookBuilder.person.First().name + "\t" + "\n";

    return addressBookBuilder; 
}

c#OutputStream

    public void SendMessage(Stream ns, byte[] msg)
    {
        byte size = (byte)msg.Length;

        try
        {
            ns.WriteByte(size);
            ns.Write(msg, 0, msg.Length);
            ns.Flush();
            ns.Close();
        }
        catch (ArgumentNullException ane)
        {
            TextBox.Text += "ArgumentNullException : {0}" + ane.ToString();
        }
        catch (Exception e)
        {
            TextBox.Text += ("Unexpected exception : {0}" + e.ToString());
        }


    }

2 个答案:

答案 0 :(得分:6)

tldr; 问题在于使用ObjectInputStream (Java) 仅适用于ObjectOutputStream (Java)生成的数据。在这种情况下,正在生成StreamCorruptedException,因为正在为流提供由<{1}} 生成 无效数据。

而是使用ObjectOutputStream (Java)来读取DataInputStream (Java)生成的数据。这两者都只支持“原始”类型。只要更正endianess used并且 sign stuffing 根据需要执行:整数,浮点数,双精度数(但不是十进制数) ),并且可以通过这种方式安全地发送字节数组。

ObjectInputStream (Java)

  

ObjectInputStream反序列化原始数据和对象以前使用ObjectOutputStream [在Java中]编写。

DataInputSteam (Java)

  

数据输入流允许应用程序从基础输入流中读取原语[..]类型

BinaryWriter (C#)

  

以二进制形式将原始类型写入流中,并支持以特定编码写入字符串。


注意:

  • DataInputSteam (Java) is big-endian,但BinaryWriter (C#) must be converted to big-endian
  • 转移BinaryWriter (C#)char/charactershortintlong和{{1}时,没有问题(除了字节顺序)数据类型,因为它们在C#和Java中具有相同的签名特性和按位表示。
  • float vs double可能会出现签名问题。值得庆幸的是,如果给出适当的byte (Java, signed) ProtocolBuffer将自动处理此
  • 由于编码差异很小,字符串可以提供额外的乐趣。

答案 1 :(得分:0)

所以使用这个c#OutputStream方法和 DataInputStream (Java)而不是ObjectOutputSteam它可以正常工作

 public void SendEndianBinaryMsg(Stream ns, byte[] msg)
    {
        byte size = (byte)msg.Length;

        try
        {

            EndianBinaryWriter writer = new EndianBinaryWriter(EndianBitConverter.Big, ns);
            writer.Write(size); 
            writer.Write(msg); 
            writer.Flush();
            ns.Close();
        }
        catch (ArgumentNullException ane)
        {
            TextBox.Text += "ArgumentNullException : {0}" + ane.ToString();
        }
        catch (Exception e)
        {
            TextBox.Text += ("Unexpected exception : {0}" + e.ToString());
        }

    }

备注:

我从MiscUtil获得 EndianBinaryWriter EndianBitConverter