정수를 바이트 배열로 변환 (Java)


137

로 변환하는 가장 빠른 방법 IntegerByte Array무엇입니까?

예 : 0xAABBCCDD => {AA, BB, CC, DD}


1
결과 바이트 배열의 형식이 중요합니까? 그것으로 무엇을 하시겠습니까?
skaffman

답변:


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] == 0xAA, result[1] == 0xBB, result[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의 바이트 배열을 할당하고 수동으로 시프트를 수행하는 것보다 할당을 수행하는 데 시간이 더 걸리는 것처럼 보입니다 ...하지만 아마도 이야기하고 있습니다. 작은 차이에 대해.
Jason S

ByteBuffer 인스턴스는 캐시 될 수 있습니다. 내부적으로는 어쨌든 시프 팅과 마스킹으로 구현됩니다.
Gregory Pakosz 2009

4
이것은 완벽하게 훌륭한 답변입니다. big-endian이 지정된 기본값이고 메소드는 "체인 가능"이고 위치 인수는 선택적이므로 모두 다음과 같이 줄어 듭니다. byte [] result = ByteBuffer.allocate (4) .putInt (0xAABBCCDD) .array ( ); 물론,이 작업을 반복적으로 수행하고 모든 결과를 함께 연결하면 (이러한 종류의 작업을 할 때 일반적 임) 단일 버퍼를 할당하고 필요한 모든 것을 반복해서 putFoo ()시킵니다- 당신이 갈 때 오프셋을 추적합니다. 정말 유용한 수업입니다.
Kevin Bourrillion 2009

서명 된 유형은 무엇입니까?
Gregory Pakosz 2016

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
바이트 순서 생각에주의하십시오
그레고리 Pakosz

1
ByteBuffer는 부호없는 int를 제공합니까?
Arun George

@Pascal ByteBuffer 사용 ByteBuffer로 시도했습니다. bb = ByteBuffer.allocate (3); 이를 위해 java.nio.BufferOverflowException이 발생합니다 .4보다 작은 값으로 작동하지 않는 이유는 무엇입니까? 설명해 주시겠습니까?
Sanjay Jain

@SanjayJain Java의 ints 크기가 32 비트 또는 4 바이트이므로 버퍼 오버 플로우 예외가 발생하므로 ByteBuffer에 최소 4 바이트의 메모리를 할당해야합니다.
충격적인

@GregoryPakosz는 바이트 순서에 관한 것입니다. ByteBuffer2 ^ 31-1보다 큰 정수를 처리하는 경우 사용하는 그의 대답 이 더 직관적입니다.
ordonezalex

31

이 기능을 사용하면 저에게 효과적입니다.

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

int를 바이트 값으로 변환합니다.


다른 답변과 비교하여 가장 중요한 비트와 더 효율적으로 관계없이 이것이 작동한다는 것도 가치가 없습니다. 또한 '>>'를 사용할 수 있습니다.
algolicious

1
이와 같은 직접 솔루션은 라이브러리 메소드를 호출하는 것보다 확실히 빠릅니다. 때로는 라이브러리 메소드 호출의 모든 추가 오버 헤드를 발생시키지 않고 몇 줄의 코드로 직접 비트를 피킹해야합니다.
David R Tribble

그리고 이것은 언어 간을 잘 변환하므로 다국어 소프트웨어 개발에 좋습니다.
코디네이터

15

구아바 를 좋아한다면 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가 전달되면 크기는 4 바이트를 초과 할 수 없습니다.

문자열에서 :

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

그러나 첫 번째 바이트가 더 높은 경우 0x7F(이 경우와 같이) BigInteger가 0x00 바이트를 배열의 시작 부분에 삽입하는 경우주의해야합니다. 양수 값과 음수 값을 구별하기 위해 필요합니다.


감사! 그러나 이것이 BigInteger이므로 int가 올바르게 래핑됩니까? 그것은 Integer.MAX_VALUE 외부의 정수이지만 여전히 4 바이트로 나타낼 수 있습니까?
Buttercup

1
이것은 확실히 빠르지는 않습니다. ;)
Peter Lawrey 2009

이것은 좋은 옵션이 아닙니다. 0x00 바이트를 추가 할뿐만 아니라 선행 0을 제거 할 수도 있습니다.
ZZ 코더

1

ByteOrder를 올바르게 처리하는 간단한 솔루션 :

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


1

안드로이드로 아주 쉽게

int i=10000;
byte b1=(byte)Color.alpha(i);
byte b2=(byte)Color.red(i);
byte b3=(byte)Color.green(i);
byte b4=(byte)Color.blue(i);

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;
}
당사 사이트를 사용함과 동시에 당사의 쿠키 정책개인정보 보호정책을 읽고 이해하였음을 인정하는 것으로 간주합니다.
Licensed under cc by-sa 3.0 with attribution required.