Java 输入流转换/签名问题

标签 java casting stream signing datainputstream

这是我在这里发表的第一篇文章,因此对可能存在的问题表示歉意。

我最近一直在使用自定义输入流,它使用字节数组来存储数据(类似于 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/

相关文章:

swift - 快速从 Scripting-Bridge 访问 SBElementArray 内容

c#:如何发布异步请求并使用 httpclient 获取流?

tomcat - 如何将 POST 数据从 servlet 读取到 ZipStream?

java - Android:一旦用户点击按钮,每 10 秒获取当前的前台 Activity

java - 代码永远无法清除的情况()

java - Maven 中的可运行 jar 以及如何使用依赖项和 list

c# - 使用 HttpWebResponse 加载页面的时间太长

java - 带有 postgresql 的串行列上的 Spring Data JPA "null value in column xxx violates not-null constraint"

c++ - static_cast<T>(...) 是编译时还是运行时?

java - 来自实现公共(public)接口(interface)的对象列表的 ClassCastException