9

我即将开始做一些需要读取字节和创建字符串的事情。正在读取的字节代表 UTF-16 字符串。所以只是为了测试一下,我想将 UTF-16 编码的简单字节数组转换为字符串。数组中的前 2 个字节必须代表字节顺序,因此必须是 0xff 0xfe 或 0xfe 0xff。所以我尝试按如下方式创建我的字节数组:

byte[] bytes = new byte[] {0xff, 0xfe, 0x52, 0x00, 0x6F, 0x00};

但是我收到了一个错误,因为 0xFF 和 0xFE 太大而无法放入一个字节(因为字节是用 Java 签名的)。更准确地说,错误是 int 无法转换为字节。我知道我可以通过强制转换从 int 显式转换为 byte 并获得所需的结果,但这不是我的问题所在。

只是为了尝试一些东西,我创建了一个字符串并调用 getBytes("UTF-16") 然后打印数组中的每个字节。输出有点混乱,因为前两个字节是 0xFFFFFFFE 0xFFFFFFFF,然后是 0x00 0x52 0x00 0x6F。(显然这里的字节序与我在上面试图创建的不同,但这并不重要)。

使用此输出,我决定尝试以相同的方式创建我的字节数组:

byte[] bytes = new byte[] {0xffffffff, 0xfffffffe, 0x52, 0x00, 0x6F, 0x00};

奇怪的是,它工作得很好。所以我的问题是,为什么 Java 允许 0xFFFFFF80 或更大的整数值在没有显式转换的情况下自动转换为字节,但等于或大于 0x80 的任何值都需要显式转换?

4

5 回答 5

10

这里要记住的关键是int在 Java 中是有符号值。当您分配0xffffffff(即2^32 -1)时,这将转换为有符号的 int 值-1-int实际上不能表示像0xffffffff正数一样大的东西。

因此对于小于 0x80 和大于 0xFFFFFF80 的值,结果int值介于 -128 和 127 之间,可以明确表示为 a byte。超出该范围的任何内容都不能,并且需要通过显式强制转换,在此过程中丢失数据。

于 2009-12-20T12:37:37.443 回答
2

如果您使用没有提示的数字(例如 1234L 表示长整数),编译器会假定为整数。该值0xffffffff是一个整数,其值-1可以在byte没有警告的情况下强制转换。

于 2009-12-20T12:33:11.353 回答
0

因为 0xffffffff 是数字 -1 而 -1 可以解释为一个字节。

于 2009-12-20T12:33:28.483 回答
0

0xff是一样的写作0x000000ff,不是0xffffffff。所以这是你的问题;整数是正数 (255),但字节(如果逐位转换)将是负数 (-1)。但是0xffffffff是 -1 作为 anint和 作为 a byte

于 2009-12-20T12:34:22.380 回答
0

因为int是有符号的,0xffffffff代表-1,0xff代表255的整数,不在-128(0x80)+127(0x7f)字节范围内。

于 2009-12-20T12:36:00.677 回答