Próbuję przekonwertować ciąg, taki jak „testing123” na postać szesnastkową w języku Java. Obecnie używam BlueJ.
A żeby przekonwertować to z powrotem, czy to jest to samo, z wyjątkiem wstecznego?
Próbuję przekonwertować ciąg, taki jak „testing123” na postać szesnastkową w języku Java. Obecnie używam BlueJ.
A żeby przekonwertować to z powrotem, czy to jest to samo, z wyjątkiem wstecznego?
Odpowiedzi:
Oto krótki sposób przekształcenia go na hex:
public String toHex(String arg) {
return String.format("%040x", new BigInteger(1, arg.getBytes(/*YOUR_CHARSET?*/)));
}
BigInteger(int,byte[])
konstruktora; w przeciwnym razie, jeśli pierwszy bajt jest ujemny, otrzymasz ujemną wartość BigInteger.
Aby mieć pewność, że szesnastka ma zawsze 40 znaków, BigInteger musi być dodatnia:
public String toHex(String arg) {
return String.format("%x", new BigInteger(1, arg.getBytes(/*YOUR_CHARSET?*/)));
}
byte[] data = { -1, 1 };
- kod w tej odpowiedzi działa dobrze, podczas gdy ten z 17 głosami za nieudany.
-1
z ciągu (zgodnie z żądaniem w przykładzie)?
byte
w tablicy ujemne wartości .
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
Liczby zakodowane w formacie szesnastkowym muszą odpowiadać kodowaniu znaków, na przykład UTF-8. Więc najpierw przekonwertuj String na bajt [] reprezentujący łańcuch w tym kodowaniu, a następnie przekonwertuj każdy bajt na szesnastkowy.
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);
}
Zastosowanie DatatypeConverter.printHexBinary()
:
public static String toHexadecimal(String text) throws UnsupportedEncodingException
{
byte[] myBytes = text.getBytes("UTF-8");
return DatatypeConverter.printHexBinary(myBytes);
}
Przykładowe użycie:
System.out.println(toHexadecimal("Hello StackOverflow"));
Wydruki:
48656C6C6F20537461636B4F766572666C6F77
Uwaga : powoduje to dodatkowe problemy w przypadku Java 9
nowszych wersji, ponieważ interfejs API nie jest domyślnie dołączony. Dla odniesienia, np. Zobacz ten GitHub
numer.
Tutaj inne rozwiązanie
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();
}
format("%02x")
więc format () zawsze używa 2 znaków. Mimo że ASCII to dwucyfrowe szesnastkowe, tj. A = 0x65
Wszystkie odpowiedzi oparte na String.getBytes () obejmują kodowanie ciągu zgodnie z Charset. Niekoniecznie dostajesz wartość szesnastkową 2-bajtowych znaków, które tworzą twój ciąg. Jeśli to, czego naprawdę chcesz, jest odpowiednikiem przeglądarki szesnastkowej, musisz uzyskać bezpośredni dostęp do znaków. Oto funkcja, której używam w moim kodzie do debugowania problemów z 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();
}
Następnie stringToHex ("testing123") daje:
0074 0065 0073 0074 0069 006e 0067 0031 0032 0033
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
Możesz powrócić hexString
w tym momencie, z zastrzeżeniem, że wiodące znaki zerowe zostaną usunięte, a wynik będzie miał nieparzystą długość, jeśli pierwszy bajt jest mniejszy niż 16. Jeśli musisz obsłużyć te przypadki, możesz dodać dodatkowy kod do wypełnienia zerami:
StringBuilder sb = new StringBuilder();
while ((sb.length() + hexString.length()) < (2 * bytes.length)) {
sb.append("0");
}
sb.append(hexString);
return sb.toString();
Aby uzyskać całkowitą wartość hex
//hex like: 0xfff7931e to int
int hexInt = Long.decode(hexString).intValue();
Zamień literę na kod szesnastkowy i kod szesnastkowy na literę.
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);
Sugerowałbym coś takiego, gdzie str
jest twój ciąg wejściowy:
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());
}
Najpierw przekonwertuj go na bajty za pomocą funkcji getBytes (), a następnie przekonwertuj na szesnastkowy, używając tego:
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();
}
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);
}
}
Konwertuj ciąg na szesnastkowy :
public String hexToString(String hex) {
return Integer.toHexString(Integer.parseInt(hex));
}
zdecydowanie jest to łatwa droga.
Korzystanie z pomocy wielu osób z wielu wątków.
Wiem, że udzielono odpowiedzi, ale chciałbym podać pełną metodę kodowania i dekodowania dla wszystkich innych w tej samej sytuacji.
Oto moje metody kodowania i dekodowania.
// 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;
}
Dużo lepiej:
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);
}
Oto kilka wzorców porównujących różne podejścia i biblioteki. Guawa pokonuje kodek Apache Commons w dekodowaniu. Commons Codec pokonuje guawę w kodowaniu. JHex pokonuje je zarówno pod względem dekodowania, jak i kodowania.
String hexString = "596f752772652077656c636f6d652e";
byte[] decoded = JHex.decodeChecked(hexString);
System.out.println(new String(decoded));
String reEncoded = JHex.encode(decoded);
Wszystko jest w jednym pliku klasy dla JHex . Możesz skopiować wklej, jeśli nie chcesz mieć kolejnej biblioteki w drzewie zależności. Zauważ też, że jest dostępny tylko jako jar Java 9, dopóki nie dowiem się, jak opublikować wiele celów wydania za pomocą Gradle i wtyczki Bintray.
Krótkim i wygodnym sposobem konwersji ciągu znaków na zapis szesnastkowy jest:
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());
}
sprawdź, czy w tym rozwiązaniu Ciąg na szesnastkowy i szesnastkowy na Ciąg odwrotnie
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();
}
Wyjście w następujący sposób:
Wyczyść tekst = testString For; 0181; with.love
Tekst szesnastkowy = 74657374537472696e6720466f723b303138313b776974682e6c6f7665
Dekodowany tekst = testString For; 0181; with.love