这是我在这里发表的第一篇文章,因此对可能存在的问题表示歉意。
我最近一直在使用自定义输入流,它使用字节数组来存储数据(类似于 ByteArrayInputStream),但对指针有更多控制。问题是由于某种原因,我的 read() 实现在值超过 127 后开始返回负数,这导致 DataInputStream 假定它是 EOF。
我将内容压缩成一个小程序来演示问题:
(分成几部分,因为我似乎无法弄清楚如何将其全部放入单个代码块中)
自定义输入流类:
class TestByteArrayInputStream 扩展了 InputStream {
字节[]数据;
{
//填充一些数据
ByteArrayOutputStream out = new ByteArrayOutputStream();
DataOutputStream doout = new DataOutputStream(out);
尝试 {
for (int i = 0; i < 256; i++) {//用值为 0-255 的 Shorts 填充数组
doout.writeShort(i);
}
} catch (可抛出的t) {
t.printStackTrace();
}
数据 = out.toByteArray();
}
int指针=0;
@覆盖
公共(public) int read() 抛出 IOException {
if (指针 >= data.length) {
指针=0;
}
返回数据[指针++];//我尝试将其转换为 char 以删除签名,并使用 Integer.valueOf,但都没有解决问题。
}
}
这是主要方法:
公共(public)类 Bugdemo {
公共(public)静态无效主(字符串[] args){
TestByteArrayInputStreamtin = new TestByteArrayInputStream();
DataInputStream din = new DataInputStream(tin);
try {//正常读取
for (int i = 0; i < 256; i++) {
System.out.println(din.readShort());
}
} catch (可抛出的t) {
System.out.println(t.toString());//避免日志记录问题
}
锡.指针 = 0;//重置到数据开头
尝试 {
for (int i = 0; i < 256; i++) {
//阅读带有添加调试功能的短代码
int ch1 =tin.read();
int ch2 =tin.read();
if ((ch1 | ch2) < 0) {
System.out.print("readshort\"eof\": ");
System.out.printf("数组中的数据为 %02X ",tin.data[tin.pointer - 2]);
System.out.printf("%02X ",tin.data[tin.pointer - 1]);
System.out.printf("但是得到%02X", ch1);
System.out.printf("%02X from read()", ch2);
System.out.println();
//抛出新的EOFException();//这是在 DataInputStream.readShort 中 if((ch1 | ch2) < 0) 之后
} 别的 {
System.out.println((短) ((ch1 << 8) + (ch2 << 0)));
}
}
} catch (可抛出的t) {
t.printStackTrace();
}
}
}
这是输出(粘贴后不会太长):http://paste.ubuntu.com/6642589/ (这里有更好的方法吗?)
重要的一点:
readshort "eof": 数组中的数据是 00 80 但从 read() 得到 00 FFFFFF80
从我的调试中,我很确定这是从数组中的字节到在 read() 中返回的 int 的转换问题,但它不应该自然地正确转换吗?如果不是,正确的方法是什么?
最佳答案
readShort
按预期工作,read
也是如此。
Java中的整数数据类型是有符号的,包括字节
。当 read
返回一个 byte
并且您按原样输出该值时,您会得到负表示。在使用 ch1 & 0xff
打印之前,您必须将其转换为具有无符号值的 int
。
关于Java 输入流转换/签名问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/20792940/