Converti intero in array di byte (Java)


137

qual è un modo veloce per convertire un Integerin un Byte Array?

per esempio 0xAABBCCDD => {AA, BB, CC, DD}


1
Ha importanza il formato dell'array di byte risultante? Cosa farai con esso?
Skaffman,

Risposte:


237

Dai un'occhiata alla classe 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();

Impostazione garantisce l'ordine byte result[0] == 0xAA, result[1] == 0xBB, result[2] == 0xCCe result[3] == 0xDD.

O in alternativa, puoi farlo manualmente:

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;
}

La ByteBufferclasse è stata progettata per compiti così sporchi. In effetti il ​​privato java.nio.Bitsdefinisce questi metodi di supporto che sono usati da 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
questo funzionerebbe bene se il bytebuffer fosse già lì ... altrimenti sembra che ci vorrebbe più tempo per fare l'allocazione, piuttosto che allocare un array di byte di lunghezza 4 e fare lo spostamento manualmente ... ma probabilmente stiamo parlando su piccole differenze.
Jason S,

L'istanza di ByteBuffer può essere memorizzata nella cache; e internamente è sicuramente implementato con spostamento e mascheramento comunque.
Gregory Pakosz,

4
Questa è una risposta perfettamente valida. Si noti che big-endian è il valore predefinito specificato e che i metodi sono "concatenabili" e l'argomento position è facoltativo, quindi tutto si riduce a: byte [] result = ByteBuffer.allocate (4) .putInt (0xAABBCCDD). ); Naturalmente, se lo fai ripetutamente e concateni tutti i risultati insieme (il che è comune quando stai facendo questo tipo di cose), alloca un singolo buffer e ripeti ripetutamente putFoo () tutte le cose di cui hai bisogno - terrà traccia dell'offset mentre procedi. È davvero una classe tremendamente utile.
Kevin Bourrillion,

cosa porta sui tipi firmati?
Gregory Pakosz,

3
Per chi non lo sa. PutInt scriverà sempre 4 byte, indipendentemente dalle dimensioni dell'intero input. Se vuoi solo 2 byte, usa putShort, ecc ...
bvdb il

36

Utilizzando BigInteger:

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

Utilizzando 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();
}

Utilizzando ByteBuffer:

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

5
presta attenzione all'ordine dei byte
Gregory Pakosz,

1
ByteBuffer rilascia un int senza segno?
Arun George,

@Pascal Usando ByteBuffer ho provato con ByteBuffer bb = ByteBuffer.allocate (3); Per questo sta dando java.nio.BufferOverflowException, non capisco perché non funziona per un valore inferiore a 4? Puoi spiegare per favore?
Sanjay Jain,

@SanjayJain Si ottiene un'eccezione di overflow del buffer poiché gli ints in Java hanno una dimensione di 32 bit o 4 byte e pertanto è necessario allocare almeno 4 byte di memoria in ByteBuffer.
scioccante

@GregoryPakosz ha ragione sull'ordine dei byte. La sua risposta usando ByteBufferè più intuitiva se hai a che fare con un int maggiore di 2 ^ 31 - 1.
ordonezalex

31

usa questa funzione funziona per me

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

traduce int in un valore byte


Inoltre, non vale nulla che funzionerà indipendentemente dal bit più significativo e più efficiente rispetto alle altre risposte. Inoltre potrebbe usare '>>'.
algolicious,

1
Una soluzione diretta come questa è sicuramente più veloce della chiamata a qualsiasi metodo di libreria. A volte devi solo giocherellare con i bit direttamente con poche righe di codice anziché incorrere in tutto il sovraccarico aggiuntivo delle chiamate al metodo della libreria.
David R Tribble,

E questo si converte bene tra le lingue, quindi è buono per lo sviluppo di software multilingue.
Il coordinatore,

15

Se ti piace Guava , puoi usare la sua Intsclasse:


Per intbyte[], utilizzare toByteArray():

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

Il risultato è {0xAA, 0xBB, 0xCC, 0xDD}.


Il suo rovescio è fromByteArray()o 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);

Il risultato è 0xAABBCCDD.


8

Puoi usare BigInteger:

Da interi:

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

L'array restituito ha le dimensioni necessarie per rappresentare il numero, quindi potrebbe essere della dimensione 1, ad esempio 1. Tuttavia, la dimensione non può essere superiore a quattro byte se viene passato un int.

Dalle stringhe:

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

Tuttavia, dovrai fare attenzione, se il primo byte è più alto 0x7F(come in questo caso), in cui BigInteger inserirà un byte 0x00 all'inizio dell'array. Ciò è necessario per distinguere tra valori positivi e negativi.


Grazie! Ma dato che si tratta di BigInteger, gli ints si avvolgeranno correttamente? Cioè numeri che sono al di fuori di Integer.MAX_VALUE ma che possono ancora essere rappresentati con solo 4 byte?
Buttercup,

1
Questo non è certamente veloce da eseguire. ;)
Peter Lawrey,

Questa non è una buona opzione. Non solo può aggiungere 0x00 byte, ma può anche eliminare gli zeri iniziali.
ZZ Coder,

1

Soluzione semplice che gestisce correttamente ByteOrder:

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


1

molto facile con Android

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

Questo ti aiuterà.

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

Può anche cambiare -

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

Ecco un metodo che dovrebbe fare il lavoro giusto.

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

È la mia soluzione:

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);
    }
}

Anche Stringy metodo:

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;
}
Utilizzando il nostro sito, riconosci di aver letto e compreso le nostre Informativa sui cookie e Informativa sulla privacy.
Licensed under cc by-sa 3.0 with attribution required.