将整数转换为字节数组(Java)


137

将转换Integer为的快速方法是Byte Array什么?

例如 0xAABBCCDD => {AA, BB, CC, DD}


1
产生的字节数组是什么格式有关系吗?您将如何处理?
skaffman 2009年

Answers:


237

看看ByteBuffer类。

ByteBuffer b = ByteBuffer.allocate(4);
//b.order(ByteOrder.BIG_ENDIAN); // optional, the initial order of a byte buffer is always BIG_ENDIAN.
b.putInt(0xAABBCCDD);

byte[] result = b.array();

设置字节顺序确保result[0] == 0xAAresult[1] == 0xBBresult[2] == 0xCCresult[3] == 0xDD

或者,您可以手动执行以下操作:

byte[] toBytes(int i)
{
  byte[] result = new byte[4];

  result[0] = (byte) (i >> 24);
  result[1] = (byte) (i >> 16);
  result[2] = (byte) (i >> 8);
  result[3] = (byte) (i /*>> 0*/);

  return result;
}

ByteBuffer班是专为尽管这样的脏手任务。实际上,私有java.nio.Bits定义了以下辅助方法ByteBuffer.putInt()

private static byte int3(int x) { return (byte)(x >> 24); }
private static byte int2(int x) { return (byte)(x >> 16); }
private static byte int1(int x) { return (byte)(x >>  8); }
private static byte int0(int x) { return (byte)(x >>  0); }

3
如果字节缓冲区已经存在,这将很好地工作...否则,似乎分配时间要比分配一个长度为4的字节数组并手动进行移位要花费更长的时间...但是我们可能正在谈论小差异。
杰森S

可以缓存ByteBuffer实例;而且在内部肯定可以通过移位和遮罩来实现。
格雷戈里·帕科斯

4
这是一个很好的答案。请注意,big-endian是指定的默认值,方法是“可链接的”,而position参数是可选的,因此全部减少为:byte [] result = ByteBuffer.allocate(4).putInt(0xAABBCCDD).array( ); 当然,如果您重复执行此操作并将所有结果连接在一起(在执行此类操作时很常见),请分配一个缓冲区,然后将所需的所有内容重复放入Foo()-它将随着您的移动跟踪偏移量。这确实是一个非常有用的课程。
凯文·布瑞里恩

它带给签名类型什么?
格雷戈里·帕科斯

3
对于谁不知道。无论输入整数的大小如何,putInt始终将写入4个字节。如果只需要2个字节,请使用putShort等
bvdb

36

使用BigInteger

private byte[] bigIntToByteArray( final int i ) {
    BigInteger bigInt = BigInteger.valueOf(i);      
    return bigInt.toByteArray();
}

使用DataOutputStream

private byte[] intToByteArray ( final int i ) throws IOException {      
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    DataOutputStream dos = new DataOutputStream(bos);
    dos.writeInt(i);
    dos.flush();
    return bos.toByteArray();
}

使用ByteBuffer

public byte[] intToBytes( final int i ) {
    ByteBuffer bb = ByteBuffer.allocate(4); 
    bb.putInt(i); 
    return bb.array();
}

5
不过要注意字节顺序
Gregory Pakosz 09年

1
ByteBuffer是否给出unsigned int吗?
阿伦·乔治

@Pascal使用ByteBuffer我尝试了ByteBuffer bb = ByteBuffer.allocate(3); 为此,它给出了java.nio.BufferOverflowException,我不明白为什么它对于小于4的值不起作用?你能解释一下吗?
Sanjay Jain 2015年

@SanjayJain因为Java中的int大小为32位或4个字节,所以会出现缓冲区溢出异常,因此需要您在ByteBuffer中分配至少4个字节的内存。
令人震惊

@GregoryPakosz关于字节顺序是正确的。ByteBuffer如果您要处理的int大于2 ^
31-1,

31

使用此功能对我有用

public byte[] toByteArray(int value) {
    return new byte[] {
            (byte)(value >> 24),
            (byte)(value >> 16),
            (byte)(value >> 8),
            (byte)value};
}

它将int转换为字节值


与其他答案相比,无论最有意义的部分还是更有效的方法,都将无法正常工作也毫无意义。也可以使用“ >>”。
吉利

1
这样的直接解决方案肯定比调用任何库方法都快。有时,您只需要用几行代码直接弄乱这些位,而不用承担库方法调用的所有额外开销。
David R Tribble,

这样可以很好地在各种语言之间进行转换,因此对于多语言软件开发非常有用。
协调员,

15

如果您喜欢Guava,则可以使用其Ints类:


对于intbyte[],使用toByteArray()

byte[] byteArray = Ints.toByteArray(0xAABBCCDD);

结果是{0xAA, 0xBB, 0xCC, 0xDD}


相反的是fromByteArray()fromBytes()

int intValue = Ints.fromByteArray(new byte[]{(byte) 0xAA, (byte) 0xBB, (byte) 0xCC, (byte) 0xDD});
int intValue = Ints.fromBytes((byte) 0xAA, (byte) 0xBB, (byte) 0xCC, (byte) 0xDD);

结果是0xAABBCCDD


8

您可以使用BigInteger

从整数:

byte[] array = BigInteger.valueOf(0xAABBCCDD).toByteArray();
System.out.println(Arrays.toString(array))
// --> {-86, -69, -52, -35 }

返回的数组具有表示数字所需的大小,因此它的大小可以为1,例如表示1。但是,如果传递一个int,则大小不能超过四个字节。

从字符串:

BigInteger v = new BigInteger("AABBCCDD", 16);
byte[] array = v.toByteArray();

但是,您需要注意,如果第一个字节较高0x7F(在这种情况下),则BigInteger将在数组的开头插入一个0x​​00字节。这需要区分正值和负值。


谢谢!但是由于这是BigInteger,int是否可以正确环绕?那是超出Integer.MAX_VALUE的整数,但仍然只能用4个字节表示吗?

1
这当然执行起来并不快。;)
Peter Lawrey 09年

这不是一个好选择。它不仅可以添加0x00字节,还可以去除前导零。
ZZ Coder 2009年

1

正确处理ByteOrder的简单解决方案:

ByteBuffer.allocate(4).order(ByteOrder.nativeOrder()).putInt(yourInt).array();



1

这将为您提供帮助。

import java.nio.ByteBuffer;
import java.util.Arrays;

public class MyClass
{
    public static void main(String args[]) {
        byte [] hbhbytes = ByteBuffer.allocate(4).putInt(16666666).array();

        System.out.println(Arrays.toString(hbhbytes));
    }
}

0

也可以移-

byte[] ba = new byte[4];
int val = Integer.MAX_VALUE;

for(byte i=0;i<4;i++)
    ba[i] = (byte)(val >> i*8);
    //ba[3-i] = (byte)(val >> i*8); //Big-endian

0

这是一种应该正确完成工作的方法。

public byte[] toByteArray(int value)
{
    final byte[] destination = new byte[Integer.BYTES];
    for(int index = Integer.BYTES - 1; index >= 0; index--)
    {
        destination[i] = (byte) value;
        value = value >> 8;
    };
    return destination;
};

0

这是我的解决方案:

public void getBytes(int val) {
    byte[] bytes = new byte[Integer.BYTES];
    for (int i = 0;i < bytes.length; i ++) {
        int j = val % Byte.MAX_VALUE;
        bytes[i] = (j == 0 ? Byte.MAX_VALUE : j);
    }
}

也是Stringy方法:

public void getBytes(int val) {
    String hex = Integer.toHexString(val);
    byte[] val = new byte[hex.length()/2]; // because byte is 2 hex chars
    for (int i = 0; i < hex.length(); i+=2)
        val[i] = Byte.parseByte("0x" + hex.substring(i, i+2), 16);
    return val;
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.