Преобразование строки в шестнадцатеричную в Java

107

Я пытаюсь преобразовать строку типа "testing123" в шестнадцатеричную форму в java. В настоящее время я использую BlueJ.

И преобразовать его обратно, это то же самое, кроме обратного?


источник

Ответы:

200

Вот краткий способ преобразовать его в шестнадцатеричный формат:

public String toHex(String arg) {
    return String.format("%040x", new BigInteger(1, arg.getBytes(/*YOUR_CHARSET?*/)));
}
Калеб Педерсон
источник
24
+1 к самому чистому образцу 3 зла, который я когда-либо видел: использование BigInteger для преобразования из байта [] ...
Эдуардо Коста
13
Любить это! Никаких циклов и никакого переворачивания битов. Я хочу дать вам 0xFF голосов :)
laher
5
чтобы обеспечить 40 символов, вы должны добавить нулевое заполнение: return String.format ("% 040x", new BigInteger (arg.getBytes (/ * YOUR_CHARSET? * /)));
Рон
4
@Kaleb У вас есть идея, можно ли преобразовать полученную строку обратно? Если да, не могли бы вы мне подсказать? Спасибо!
artaxerxe
1
Вы должны использовать BigInteger(int,byte[])конструктор; в противном случае, если первый байт отрицательный, вы получите отрицательное значение BigInteger.
Joni
61

Чтобы шестнадцатеричный код всегда был длиной 40 символов, значение BigInteger должно быть положительным:

public String toHex(String arg) {
  return String.format("%x", new BigInteger(1, arg.getBytes(/*YOUR_CHARSET?*/)));
}
Джос Теувен
источник
1
Этот метод действительно правильный. Попробуйте byte[] data = { -1, 1 };- код в этом ответе работает нормально, тогда как с 17 голосами за него не удается.
hudolejev 01
1
Можно ли получить байт со значением -1из строки (как было запрошено в примере)?
Калеб Педерсон
@KalebPederson Да. Это даже не очень сложно. . Если выбранная вами кодировка когда-либо использует самый старший бит любого символа (скажем, как UTF- * do), у вас есть отрицательный bytes в вашем массиве.
Иск Фонд Моники
45
import org.apache.commons.codec.binary.Hex;
...

String hexString = Hex.encodeHexString(myString.getBytes(/* charset */));

http://commons.apache.org/codec/apidocs/org/apache/commons/codec/binary/Hex.html

Джошуа Суинк
источник
3
Интересно, если не хотите изобретать велосипед.
Federico Zancan
3
@MelNicholson в Hex есть функция decodeHex для перехода к байту []. Вам нужно использовать это, потому что ничто не гарантирует, что случайная строка HEX может быть преобразована в строку в вашей кодировке.
BxlSofty
19

Числа, которые вы кодируете в шестнадцатеричном формате, должны представлять некоторую кодировку символов, например UTF-8. Поэтому сначала преобразуйте String в byte [], представляющий строку в этой кодировке, а затем преобразуйте каждый байт в шестнадцатеричный.

public static String hexadecimal(String input, String charsetName) throws UnsupportedEncodingException {
    if (input == null) throw new NullPointerException();
    return asHex(input.getBytes(charsetName));
}

private static final char[] HEX_CHARS = "0123456789abcdef".toCharArray();

public static String asHex(byte[] buf)
{
    char[] chars = new char[2 * buf.length];
    for (int i = 0; i < buf.length; ++i)
    {
        chars[2 * i] = HEX_CHARS[(buf[i] & 0xF0) >>> 4];
        chars[2 * i + 1] = HEX_CHARS[buf[i] & 0x0F];
    }
    return new String(chars);
}
Стивен Денн
источник
Это интересное решение, которое поражает суть цифрового представления данных. Не могли бы вы объяснить, что вы делаете и что представляют собой «магические числа» в вашем решении? Новичок может не знать, что означает оператор >>>, почему мы используем побитовое и & вместе с маской 0xF0 или почему массив символов имеет размер [2 * buf.length].
Борис
16

Использование DatatypeConverter.printHexBinary():

public static String toHexadecimal(String text) throws UnsupportedEncodingException
{
    byte[] myBytes = text.getBytes("UTF-8");

    return DatatypeConverter.printHexBinary(myBytes);
}

Пример использования:

System.out.println(toHexadecimal("Hello StackOverflow"));

Печать:

48656C6C6F20537461636B4F766572666C6F77

Примечание : это вызывает небольшие дополнительные проблемы с Java 9более новой версией, поскольку API не включен по умолчанию. Для справки, например, см. Этот GitHub выпуск.

BullyWiiPlaza
источник
11

Вот другое решение

public static String toHexString(byte[] ba) {
    StringBuilder str = new StringBuilder();
    for(int i = 0; i < ba.length; i++)
        str.append(String.format("%x", ba[i]));
    return str.toString();
}

public static String fromHexString(String hex) {
    StringBuilder str = new StringBuilder();
    for (int i = 0; i < hex.length(); i+=2) {
        str.append((char) Integer.parseInt(hex.substring(i, i + 2), 16));
    }
    return str.toString();
}
Jordeu
источник
3
Хорошо, но я бы использовал, format("%02x")поэтому format () всегда использует 2 символа. Несмотря на то, что ASCII - это двузначный шестнадцатеричный код, то есть A = 0x65
Майк Джонс
7

Все ответы, основанные на String.getBytes (), включают кодирование вашей строки в соответствии с Charset. Вы не обязательно получите шестнадцатеричное значение двухбайтовых символов , составляющих вашу строку. Если то, что вы действительно хотите, является эквивалентом шестнадцатеричного средства просмотра, вам необходимо получить доступ к символам напрямую. Вот функция, которую я использую в своем коде для отладки проблем с Unicode:

static String stringToHex(String string) {
  StringBuilder buf = new StringBuilder(200);
  for (char ch: string.toCharArray()) {
    if (buf.length() > 0)
      buf.append(' ');
    buf.append(String.format("%04x", (int) ch));
  }
  return buf.toString();
}

Затем stringToHex ("testing123") даст вам:

0074 0065 0073 0074 0069 006e 0067 0031 0032 0033
Богдан Калмак
источник
Это нормально, если вы хотите увидеть внутреннее представление символов Java, то есть UTF-16, конкретное представление Unicode.
Джонатан Розенн,
5

Чтобы получить целочисленное значение шестнадцатеричного

        //hex like: 0xfff7931e to int
        int hexInt = Long.decode(hexString).intValue();
TouchBoarder
источник
4
byte[] bytes = string.getBytes(CHARSET); // you didn't say what charset you wanted
BigInteger bigInt = new BigInteger(bytes);
String hexString = bigInt.toString(16); // 16 is the radix

Вы можете вернуться hexStringв этот момент с оговоркой, что ведущие нулевые символы будут удалены, и результат будет иметь нечетную длину, если первый байт меньше 16. Если вам нужно обработать эти случаи, вы можете добавить дополнительный код. для заполнения нулями:

StringBuilder sb = new StringBuilder();
while ((sb.length() + hexString.length()) < (2 * bytes.length)) {
  sb.append("0");
}
sb.append(hexString);
return sb.toString();
Лоуренс Гонсалвес
источник
4

Преобразование буквы в шестнадцатеричный код и шестнадцатеричный код в букву.

        String letter = "a";
    String code;
    int decimal;

    code = Integer.toHexString(letter.charAt(0));
    decimal = Integer.parseInt(code, 16);

    System.out.println("Hex code to " + letter + " = " + code);
    System.out.println("Char to " + code + " = " + (char) decimal);
Маркус Беккер
источник
4

Я бы предложил что-то вроде этого, где strваша строка ввода:

StringBuffer hex = new StringBuffer();
char[] raw = tokens[0].toCharArray();
for (int i=0;i<raw.length;i++) {
    if     (raw[i]<=0x000F) { hex.append("000"); }
    else if(raw[i]<=0x00FF) { hex.append("00" ); }
    else if(raw[i]<=0x0FFF) { hex.append("0"  ); }
    hex.append(Integer.toHexString(raw[i]).toUpperCase());
}
Родион
источник
Спасибо за исправление, Software Monkey. Я очень устал, когда писал ответ, и мой тест на «raw [i] <= 9» явно недостаточен.
Родион
1
Это действительно хорошо работает, есть ли способ снова вернуть сгенерированный шестнадцатеричный код обратно в строку?
1
Где в этом str?
Вишванат Лекшманан
3

Чтобы пойти другим путем (шестнадцатеричный в строку), вы можете использовать

public String hexToString(String hex) {
    return new String(new BigInteger(hex, 16).toByteArray());
}
Джибби
источник
2

Сначала преобразуйте его в байты с помощью функции getBytes (), а затем преобразуйте его в шестнадцатеричный, используя это:

private static String hex(byte[] bytes) {
    StringBuilder sb = new StringBuilder();
    for (int i=0; i<bytes.length; i++) {
        sb.append(String.format("%02X ",bytes[i]));
    }
    return sb.toString();
}
user2475511
источник
2

Использование поддержки нескольких людей из нескольких потоков ..

Я знаю, что на это был дан ответ, но я хотел бы дать полный метод кодирования и декодирования для любых других в той же ситуации.

Вот мои методы кодирования и декодирования ..

// Global Charset Encoding
public static Charset encodingType = StandardCharsets.UTF_8;

// Text To Hex
public static String textToHex(String text)
{
    byte[] buf = null;
    buf = text.getBytes(encodingType);
    char[] HEX_CHARS = "0123456789abcdef".toCharArray();
    char[] chars = new char[2 * buf.length];
    for (int i = 0; i < buf.length; ++i)
    {
        chars[2 * i] = HEX_CHARS[(buf[i] & 0xF0) >>> 4];
        chars[2 * i + 1] = HEX_CHARS[buf[i] & 0x0F];
    }
    return new String(chars);
}

// Hex To Text
public static String hexToText(String hex)
{
    int l = hex.length();
    byte[] data = new byte[l / 2];
    for (int i = 0; i < l; i += 2)
    {
        data[i / 2] = (byte) ((Character.digit(hex.charAt(i), 16) << 4)
            + Character.digit(hex.charAt(i + 1), 16));
    }
    String st = new String(data, encodingType);
    return st;
}
Гаджет Гуру
источник
Ницца! Только в строке 13 ">>>" должно быть ">>"
spikeyang
0

Намного лучше:

public static String fromHexString(String hex, String sourceEncoding ) throws  IOException{
    ByteArrayOutputStream bout = new ByteArrayOutputStream();
    byte[] buffer = new byte[512];
    int _start=0;
    for (int i = 0; i < hex.length(); i+=2) {
        buffer[_start++] = (byte)Integer.parseInt(hex.substring(i, i + 2), 16);
        if (_start >=buffer.length || i+2>=hex.length()) {
            bout.write(buffer);
            Arrays.fill(buffer, 0, buffer.length, (byte)0);
            _start  = 0;
        }
    }

    return  new String(bout.toByteArray(), sourceEncoding);
}
Дмитрий Бочаров
источник
0
import java.io.*;
import java.util.*;

public class Exer5{

    public String ConvertToHexadecimal(int num){
        int r;
        String bin="\0";

        do{
            r=num%16;
            num=num/16;

            if(r==10)
            bin="A"+bin;

            else if(r==11)
            bin="B"+bin;

            else if(r==12)
            bin="C"+bin;

            else if(r==13)
            bin="D"+bin;

            else if(r==14)
            bin="E"+bin;

            else if(r==15)
            bin="F"+bin;

            else
            bin=r+bin;
        }while(num!=0);

        return bin;
    }

    public int ConvertFromHexadecimalToDecimal(String num){
        int a;
        int ctr=0;
        double prod=0;

        for(int i=num.length(); i>0; i--){

            if(num.charAt(i-1)=='a'||num.charAt(i-1)=='A')
            a=10;

            else if(num.charAt(i-1)=='b'||num.charAt(i-1)=='B')
            a=11;

            else if(num.charAt(i-1)=='c'||num.charAt(i-1)=='C')
            a=12;

            else if(num.charAt(i-1)=='d'||num.charAt(i-1)=='D')
            a=13;

            else if(num.charAt(i-1)=='e'||num.charAt(i-1)=='E')
            a=14;

            else if(num.charAt(i-1)=='f'||num.charAt(i-1)=='F')
            a=15;

            else
            a=Character.getNumericValue(num.charAt(i-1));
            prod=prod+(a*Math.pow(16, ctr));
            ctr++;
        }
        return (int)prod;
    }

    public static void main(String[] args){

        Exer5 dh=new Exer5();
        Scanner s=new Scanner(System.in);

        int num;
        String numS;
        int choice;

        System.out.println("Enter your desired choice:");
        System.out.println("1 - DECIMAL TO HEXADECIMAL             ");
        System.out.println("2 - HEXADECIMAL TO DECIMAL              ");
        System.out.println("0 - EXIT                          ");

        do{
            System.out.print("\nEnter Choice: ");
            choice=s.nextInt();

            if(choice==1){
                System.out.println("Enter decimal number: ");
                num=s.nextInt();
                System.out.println(dh.ConvertToHexadecimal(num));
            }

            else if(choice==2){
                System.out.println("Enter hexadecimal number: ");
                numS=s.next();
                System.out.println(dh.ConvertFromHexadecimalToDecimal(numS));
            }
        }while(choice!=0);
    }
}
Ровена Хименес
источник
0
new BigInteger(1, myString.getBytes(/*YOUR_CHARSET?*/)).toString(16)
ультраон
источник
0

Вот несколько тестов, сравнивающих разные подходы и библиотеки. Guava превосходит кодек Apache Commons при декодировании. Commons Codec превосходит Guava при кодировании. И JHex превосходит их как по декодированию, так и по кодированию.

Пример JHex

String hexString = "596f752772652077656c636f6d652e";
byte[] decoded = JHex.decodeChecked(hexString);
System.out.println(new String(decoded));
String reEncoded = JHex.encode(decoded);

Все находится в одном файле класса для JHex . Не стесняйтесь копировать пасту, если вам не нужна еще одна библиотека в вашем дереве зависимостей. Также обратите внимание, что он доступен только как Java 9 jar, пока я не выясню, как опубликовать несколько целей выпуска с помощью Gradle и плагина Bintray.

jamespedwards42
источник
0

Преобразовать строку в шестнадцатеричный :

public String hexToString(String hex) {
    return Integer.toHexString(Integer.parseInt(hex));
}

определенно это легкий путь.

Jorgesys
источник
Это не выход. Вопрос заключается в том, как получить шестнадцатеричное представление содержимого произвольной строки, и в частности, в качестве примера предоставлено "testing123" .
skomisa
0

проверьте это решение для String в шестнадцатеричный и шестнадцатеричный в String, наоборот

public class TestHexConversion {
public static void main(String[] args) {
    try{
        String clearText = "testString For;0181;with.love";
        System.out.println("Clear Text  = " + clearText);
        char[] chars = clearText.toCharArray();
        StringBuffer hex = new StringBuffer();
        for (int i = 0; i < chars.length; i++) {
            hex.append(Integer.toHexString((int) chars[i]));
        }
        String hexText = hex.toString();
        System.out.println("Hex Text  = " + hexText);
        String decodedText = HexToString(hexText);
        System.out.println("Decoded Text = "+decodedText);
    } catch (Exception e){
        e.printStackTrace();
    }
}

public static String HexToString(String hex){

      StringBuilder finalString = new StringBuilder();
      StringBuilder tempString = new StringBuilder();

      for( int i=0; i<hex.length()-1; i+=2 ){
          String output = hex.substring(i, (i + 2));
          int decimal = Integer.parseInt(output, 16);
          finalString.append((char)decimal);
          tempString.append(decimal);
      }
    return finalString.toString();
}

Вывод следующий:

Очистить текст = testString For; 0181; with.love

Hex Text = 74657374537472696e6720466f723b303138313b776974682e6c6f7665

Расшифрованный текст = testString For; 0181; with.love

Нитин Упадхьяй
источник
-1

Короткий и удобный способ преобразовать строку в ее шестнадцатеричное представление:

public static void main(String... args){
String str = "Hello! This is test string.";
char ch[] = str.toCharArray();
StringBuilder sb = new StringBuilder();
    for (int i = 0; i < ch.length; i++) {
        sb.append(Integer.toHexString((int) ch[i]));
    }
    System.out.println(sb.toString());
}
Амит Самуэль
источник