Як перетворити байт у його двійковий рядок


92

Наприклад, біти в байті - Bце 10000010те, як я можу присвоїти біти рядку strбуквально, тобто str = "10000010".

Редагувати

Я прочитав байт із двійкового файлу та зберігав у байтовому масиві B. Я використовую System.out.println(Integer.toBinaryString(B[i])). проблема в тому

(а) коли біти починаються з (крайнього лівого) 1, вихід не є правильним, оскільки він перетворюється B[i]на від'ємне значення int.

(b) якщо біти починаються з 0, вихід ігнорує 0, наприклад, припустимо, B[0]має 00000001, вихід 1замість00000001


2
Я збентежений; це фокус?
Dave Newton

1
Ви запитуєте, як перетворити a byteна рядок в основі 2?
Слакс

Я просто додав відповідь на інший потік для цього (перетворення значення в рядок двійкових цифр) , яка працює на Boolean, Byte, Short, Char, Int, і Long. stackoverflow.com/a/54950845/501113
chaotic3quilibrium

String # Format () міг би це впоратись, якщо ви сказали йому використовувати ширину 8. Так само System.out.printf ().
NomadMaker

Відповіді:


170

Використання Integer#toBinaryString():

byte b1 = (byte) 129;
String s1 = String.format("%8s", Integer.toBinaryString(b1 & 0xFF)).replace(' ', '0');
System.out.println(s1); // 10000001

byte b2 = (byte) 2;
String s2 = String.format("%8s", Integer.toBinaryString(b2 & 0xFF)).replace(' ', '0');
System.out.println(s2); // 00000010

ДЕМО .


Я спробував цей метод. У моєму випадку я читаю байт із двійкового файлу і зберігаю в байтовому масиві B. Я використовую System.out.println(Integer.toBinaryString(B[i])). Коли я використовую ці методи, проблема полягає в (а), коли біти починаються з (крайнього лівого) 1, вихід не правильний, оскільки він перетворюється B[i]на від'ємне значення int. (b) якщо біти починаються з 0, вихід ігнорується 0, наприклад, припустимо, B[0]має 00000001, вихід 1замість00000001
Шон

1
@Sean: а) трапляється, оскільки a byteв Java є цілим числом доповнення 8-бітовим двома знаками . Його мінімальне значення -128 (2 ^ 8), а максимальне - 127; б) Ви можете легко це виправити, скориставшись цим, String.format("%8s", Integer.toBinaryString(b)).replace(' ', '0')щоб зліва вкласти отриманий рядок нулями.
João Silva

1
@ João: дякую за вашу пораду. Чи маєте ви уявлення про те, як звертатися до (а), як зберігати оригінальний бітовий формат (починається з 1) у рядку?
Шон

1
@Sean: Так, просто &це з 0xFF.
João Silva

12
@Sean: в & 0xFFосновному перетворює a signed byteна unsigned integer. Наприклад, -129як ви вже сказали, представлено 11111111111111111111111110000001. У цьому випадку ви в основному хочете перші (найменш значущі) 8 біт, тому ви І ( &) це за допомогою 0xFF( 00000000000000000000000011111111), ефективно очищаючи одиниці ліворуч, про які ми не дбаємо, залишаючи просто 10000001.
João Silva

33

Я цим користувався. Подібна ідея до інших відповідей, але ніде не бачив точного підходу :)

System.out.println(Integer.toBinaryString((b & 0xFF) + 0x100).substring(1));

0xFFдорівнює 255 або 11111111(максимальне значення для непідписаного байта). 0x100дорівнює 256, або100000000

Збільшує &байт до цілого числа. На той момент це може бути що завгодно від 0- 255( 00000000до 11111111, я виключив провідні 24 біти). + 0x100і .substring(1)переконайтеся, що будуть перші нулі.

Я приурочив його порівняно з відповіддю Джоау Сільви , і це в 10 разів швидше. http://ideone.com/22DDK1 Я не включив відповідь Пшемо, оскільки він не накладається належним чином.


привіт! отримав запитання щодо цього. У мене є рядок представлення Base64 у форматі PDF, мені потрібно перетворити його у двійковий файл. В основному, Base64-> байт-> двійковий. Чи спрацює цей код?
Сід

Що саме робить + 0x100? Ви додаєте 256 до отриманого цілого числа, але чому?
Коннер Дассен

1
@ConnerDassen Це гарантує, що двійковий рядок дорівнює 0. Наприклад, якщо bє 1, без + 0x100вас просто отримають "1"як ваш рядок. Додаючи 1, ви отримуєте 100000001, і якщо ви берете підрядок, ігноруючи перший символ, ви отримаєте правильний "00000001". Якщо ви не хочете, щоб рядок був заповнений, ви можете просто використовувати Integer.toBinaryString(b & 0xff). Виправлено & 0xffпроблеми з доповненням негативу / двох
Raekye

8

Це те, що ви шукаєте?

перетворення з рядка в байт

byte b = (byte)(int)Integer.valueOf("10000010", 2);
System.out.println(b);// output -> -126

перетворення з байта в рядок

System.out.println(Integer.toBinaryString((b+256)%256));// output -> "10000010"

Або, як сказав Жоао Сільва у своєму коментарі, щоб додати провідний, 0ми можемо відформатувати рядок довжиною 8 і замінити результуючі пробіли на початку нулем, тому у випадку рядка, як " 1010"ми отримаємо"00001010"

System.out.println(String.format("%8s", Integer.toBinaryString((b + 256) % 256))
                         .replace(' ', '0'));

5

Ви можете перевірити кожен біт байта, а потім додати до рядка 0 або 1. Ось маленький допоміжний метод, який я написав для тестування:

public static String byteToString(byte b) {
    byte[] masks = { -128, 64, 32, 16, 8, 4, 2, 1 };
    StringBuilder builder = new StringBuilder();
    for (byte m : masks) {
        if ((b & m) == m) {
            builder.append('1');
        } else {
            builder.append('0');
        }
    }
    return builder.toString();
}

3

Отримайте кожен біт байта і перетворіть у рядок. Скажімо, байт має 8 бітів, і ми можемо отримати їх по одному за допомогою бітового переміщення. Наприклад, ми переміщуємо другий біт байта 6 бітів вправо, другий біт нарешті біт 8 бітів, а потім (&) з 0x0001 для очищення фронтальних бітів.

public static String getByteBinaryString(byte b) {
    StringBuilder sb = new StringBuilder();
    for (int i = 7; i >= 0; --i) {
        sb.append(b >>> i & 1);
    }
    return sb.toString();
}

Не могли б ви відредагувати свою відповідь, щоб пояснити, чому цей код відповідає на питання? Відповіді лише на код не рекомендується , оскільки вони не вчать розв’язанню.
DavidPostill,

2

Цей код продемонструє, як java int можна розділити на 4 послідовні байти. Потім ми можемо перевірити кожен байт, використовуючи методи Java порівняно з низьким рівнем опитування байтів / бітів.

Це очікуваний результат при запуску коду нижче:

[Input] Integer value: 8549658

Integer.toBinaryString: 100000100111010100011010
Integer.toHexString: 82751a
Integer.bitCount: 10

Byte 4th Hex Str: 0
Byte 3rd Hex Str: 820000
Byte 2nd Hex Str: 7500
Byte 1st Hex Str: 1a

(1st + 2nd + 3rd + 4th (int(s)) as Integer.toHexString: 82751a
(1st + 2nd + 3rd + 4th (int(s)) ==  Integer.toHexString): true

Individual bits for each byte in a 4 byte int:
00000000 10000010 01110101 00011010

Ось код для запуску:

public class BitsSetCount
{
    public static void main(String[] args) 
    {
        int send = 8549658;

        System.out.println( "[Input] Integer value: " + send + "\n" );
        BitsSetCount.countBits(  send );
    }

    private static void countBits(int i) 
    {
        System.out.println( "Integer.toBinaryString: " + Integer.toBinaryString(i) );
        System.out.println( "Integer.toHexString: " + Integer.toHexString(i) );
        System.out.println( "Integer.bitCount: "+ Integer.bitCount(i) );

        int d = i & 0xff000000;
        int c = i & 0xff0000;
        int b = i & 0xff00;
        int a = i & 0xff;

        System.out.println( "\nByte 4th Hex Str: " + Integer.toHexString(d) );
        System.out.println( "Byte 3rd Hex Str: " + Integer.toHexString(c) );
        System.out.println( "Byte 2nd Hex Str: " + Integer.toHexString(b) );
        System.out.println( "Byte 1st Hex Str: " + Integer.toHexString(a) );

        int all = a+b+c+d;
        System.out.println( "\n(1st + 2nd + 3rd + 4th (int(s)) as Integer.toHexString: " + Integer.toHexString(all) );

        System.out.println("(1st + 2nd + 3rd + 4th (int(s)) ==  Integer.toHexString): " + 
                Integer.toHexString(all).equals(Integer.toHexString(i) ) );

        System.out.println( "\nIndividual bits for each byte in a 4 byte int:");

        /*
         * Because we are sending the MSF bytes to a method
         * which will work on a single byte and print some
         * bits we are generalising the MSF bytes
         * by making them all the same in terms of their position
         * purely for the purpose of printing or analysis
         */
        System.out.print( 
                    getBits( (byte) (d >> 24) ) + " " + 
                    getBits( (byte) (c >> 16) ) + " " + 
                    getBits( (byte) (b >> 8) ) + " " + 
                    getBits( (byte) (a >> 0) ) 
        );


    }

    private static String getBits( byte inByte )
    {
        // Go through each bit with a mask
        StringBuilder builder = new StringBuilder();
        for ( int j = 0; j < 8; j++ )
        {
            // Shift each bit by 1 starting at zero shift
            byte tmp =  (byte) ( inByte >> j );

            // Check byte with mask 00000001 for LSB
            int expect1 = tmp & 0x01; 

            builder.append(expect1);
        }
        return ( builder.reverse().toString() );
    }

}


2

На жаль, я знаю, що це трохи пізно ... Але у мене набагато простіший спосіб ... До двійкового рядка:

//Add 128 to get a value from 0 - 255
String bs = Integer.toBinaryString(data[i]+128);
bs = getCorrectBits(bs, 8);

метод getCorrectBits:

private static String getCorrectBits(String bitStr, int max){
    //Create a temp string to add all the zeros
    StringBuilder sb = new StringBuilder();
    for(int i = 0; i < (max - bitStr.length()); i ++){
        sb.append("0");
    }

    return sb.toString()+ bitStr;
}

1
String byteToBinaryString(byte b){
    StringBuilder binaryStringBuilder = new StringBuilder();
    for(int i = 0; i < 8; i++)
        binaryStringBuilder.append(((0x80 >>> i) & b) == 0? '0':'1');
    return binaryStringBuilder.toString();
}

1

Ви можете працювати з BigInteger, як показано нижче, особливо, особливо якщо у вас 256 біт або довше:

String string = "10000010";
BigInteger biStr = new BigInteger(string, 2);

System.out.println("binary: " + biStr.toString(2));
System.out.println("hex: " + biStr.toString(16));
System.out.println("dec: " + biStr.toString(10));

Інший приклад, який приймає байти:

String string = "The girl on the red dress.";

byte[] byteString = string.getBytes(Charset.forName("UTF-8"));
System.out.println("[Input String]: " + string);
System.out.println("[Encoded String UTF-8]: " + byteString);

BigInteger biStr = new BigInteger(byteString);
System.out.println("binary: " + biStr.toString(2)); // binary
System.out.println("hex: " + biStr.toString(16));   // hex or base 16
System.out.println("dec: " + biStr.toString(10));  // this is base 10

Результат:

[Input String]: The girl on the red dress.
[Encoded String UTF-8]: [B@70dea4e

binary: 101010001101000011001010010000001100111011010010111001001101100001000000110111101101110001000000111010001101000011001010010000001110010011001010110010000100000011001000111001001100101011100110111001100101110
hex: 546865206769726c206f6e20746865207265642064726573732e

Ви також можете працювати над перетворенням двійкового формату в байтний формат

try {
   System.out.println("binary to byte: " + biStr.toString(2).getBytes("UTF-8"));
} catch (UnsupportedEncodingException e) {e.printStackTrace();}

Примітка: Для форматування рядків для вашого бінарного формату ви можете використовувати наведений нижче зразок

String.format("%256s", biStr.toString(2).replace(' ', '0'));  // this is for the 256 bit formatting

1

Проста відповідь може бути такою:

System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0})); // 0
System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 1})); // 1
System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 0, 0, 0, 0, 0, 1, 0})); // 256
System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 0, 0, 0, 0, 1, 0, 0})); // 65536
System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 0, 0, 0, 1, 0, 0, 0})); // 16777216
System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 0, 0, 1, 0, 0, 0, 0})); // 4294967296
System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 0, 1, 0, 0, 0, 0, 0})); // 1099511627776
System.out.println(new BigInteger(new byte[]{0, 0, 0, 0, 1, 0, 0, 0, 0, 0, 0})); // 281474976710656
System.out.println(new BigInteger(new byte[]{0, 0, 0, 1, 0, 0, 0, 0, 0, 0, 0})); // 72057594037927936
System.out.println(new BigInteger(new byte[]{0, 0, 1, 0, 0, 0, 0, 0, 0, 0, 0})); // 18446744073709551616
System.out.println(new BigInteger(new byte[]{0, 1, 0, 0, 0, 0, 0, 0, 0, 0, 0})); // 4722366482869645213696
System.out.println(new BigInteger(new byte[]{1, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0})); // 1208925819614629174706176
System.out.println(Long.MAX_VALUE);                                              // 9223372036854775807

0

Ми всі знаємо, що Java не пропонує нічого подібного до ключового слова без підпису. Більше того, byteпримітив відповідно до специфікації Java представляє значення між −128і 127. Наприклад, якщо byteє castна intJava буде інтерпретувати перший , bitяк signі розширення використання знака.

Тоді, як перетворити байт, більший ніж 127у його двійковий рядок ??

Ніщо не заважає переглядати byteпросто 8-біт і інтерпретувати ці біти як значення між 0і 255. Крім того, потрібно пам’ятати, що ви нічого не можете зробити, щоб нав'язати свою інтерпретацію чужому методу. Якщо метод приймає a byte, тоді цей метод приймає значення між −128і, 127якщо явно не вказано інше.

Тож найкращий спосіб вирішити це - перетворити byteзначення у intзначення, викликавши Byte.toUnsignedInt()метод або відливши його як intпримітив (int) signedByte & 0xFF. Ось приклад:

public class BinaryOperations
{
    public static void main(String[] args)
    {
        byte forbiddenZeroBit = (byte) 0x80;

        buffer[0] = (byte) (forbiddenZeroBit & 0xFF);
        buffer[1] = (byte) ((forbiddenZeroBit | (49 << 1)) & 0xFF);
        buffer[2] = (byte) 96;
        buffer[3] = (byte) 234;

        System.out.println("8-bit header:");
        printBynary(buffer);
    }

    public static void printBuffer(byte[] buffer)
    {
        for (byte num : buffer) {
            printBynary(num);
        }
    }

    public static void printBynary(byte num)
    {
        int aux = Byte.toUnsignedInt(num);
        // int aux = (int) num & 0xFF; 
        String binary = String.format("%8s', Integer.toBinaryString(aux)).replace(' ', '0');
        System.out.println(binary);
    }
}

Вихідні дані

8-bit header:
10000000
11100010
01100000
11101010

-1

Просто гадаю тут, але якщо у вас є байт, то не могли б ви просто викликати toString () на об’єкті, щоб отримати значення? Або, поглянувши на api , використовуючи byteValue ()?

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.