3

我有一个Java类

public class MsgLayout{
int field1;
String field2;
long field3;
}

我必须将此对象写为 Socket 输出流中的字节数组。这三个字段(实例变量)有一个布局。即field1必须占用1个字节,field2必须占用4个字节,field3必须占用8个字节。

ByteBuffer bbf = ByteBuffer.allocate(TOTAL_SIZE);
bbf.put(Integer.toString(this.getField1()).getBytes(), 0, FIELD1_SIZE);
bbf.position(FIELD2_OFFSET);
bbf.put(Long.toString(this.getField2()).getBytes(), 0, FIELD2_SIZE);
bbf.position(FIELD3_OFFSET);
bbf.put(Long.toString(this.getField3()).getBytes(), 0, FIELD3_SIZE);
byte[] msg = bbf.array();

使用上面的代码,我试图根据所需的大小来适应字节数组中的每个字段。但我得到IndexOutOfBoundException 简而言之,问题在于如何将字段适应布局定义的大小。例如FIELD1_OFFSET = 0, FIELD1_SIZE=1, FIELD2_OFFSET=1, FIELD2_SIZE=4, FIELD3_OFFSET=5, FIELD3_SIZE=8. 现在当我转换field1成字符串时,转换成字节 [] 时它不适合 1 个字节。如果我不转换为字符串,并使用 putInt(int) 它将 4 个字节写入结果字节数组。

4

2 回答 2

2

您的代码当前正在做的是将您的数字字段编码为字符串,然后输出这些字符的字节。

我建议使用 DataOutputStream 类来包装你的 SocketOutput 流并这样写你的二进制数据:

DataOutput output = new DataOutputStream(socketOutputStream);

int field1 = 1;
String field2 = "Hello";
long field3 = 5000000000L;

output.writeByte(field1);
output.writeBytes(field2.substring(0, 3));
output.writeLong(field3);

这段代码中有几个假设。首先,我假设对于字段 2,您希望将 4 个字符序列化为每个字符。如果您想使用 UTF-8 之类的东西进行任何多字节编码,那么您需要做一些不同的事情。其次,我假设字段 2 总是至少有 4 个字符。

于 2011-09-28T14:52:06.250 回答
0

field1可能只有一个字节的数据,但它的字符串表示将是一个或多个字符(例如"0",,,"63""127"。String 中的每个字符实际上都是一个char(两个字节的值)。所以我希望一个字节的数据在经过 byte->String->byte[] 转换时膨胀到 2 到 6 个字节的数据。

于 2011-09-28T14:48:26.157 回答