问题
我正在尝试从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)
说实话,我有点不知所措。任何帮助表示赞赏。谢谢!
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());
}
}
答案 0 :(得分:6)
tldr; 问题在于使用ObjectInputStream (Java)
仅适用于与ObjectOutputStream (Java)
生成的数据。在这种情况下,正在生成StreamCorruptedException
,因为正在为流提供由<{1}} 生成 的无效数据。
而是使用ObjectOutputStream (Java)
来读取DataInputStream (Java)
生成的数据。这两者都只支持“原始”类型。只要更正endianess 为used并且 sign stuffing 根据需要执行:整数,浮点数,双精度数(但不是十进制数) ),并且可以通过这种方式安全地发送字节数组。
ObjectInputStream反序列化原始数据和对象以前使用ObjectOutputStream [在Java中]编写。
数据输入流允许应用程序从基础输入流中读取原语[..]类型。
以二进制形式将原始类型写入流中,并支持以特定编码写入字符串。
注意:
BinaryWriter (C#)
,char/character
,short
,int
,long
和{{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 。