JAVA⼗六进制与字符串的转换
笔者前⼏⽇在开服过程中需要将字符串转化成为16进制的字符串,在⽹上到了⼀些⽅法尝试之后,均发现存在⼀个问题-->字符串转为16进制后再转回来,英⽂正常,中⽂出现乱码
经过考虑决定通过以下⽅式进⾏解决:
1)在将字符串转为16进制之前先进⾏⼀次转化,先将其转化成为Unicode编码(相当于把中⽂⽤英⽂字符代替),在转化成为16进制
2)相反的,在⼗六进制转换为字符串后的得到的是Unicode编码,此时再将Unicode编码解码即可获取原始字符串
代码如下:
/**
* 字符串转换unicode
*/
public static String string2Unicode(String string) {
StringBuffer unicode = new StringBuffer();
for (int i = 0; i < string.length(); i++) {
// 取出每⼀个字符
char c = string.charAt(i);
// 转换为unicode
unicode.append("\\u" + HexString(c));
}
String();
}
*字符串转为16进制
/
**
* 字符串转化成为16进制字符串
* @param s
* @return
*/
public static String strTo16(String s) {
String str = "";
for (int i = 0; i < s.length(); i++) {
int ch = (int) s.charAt(i);
String s4 = HexString(ch);
str = str + s4;
}
return str;
}
*16进制转为字符串
/**
* 16进制转换成为string类型字符串
* @param s
* @return
*/
public static String hexStringToString(String s) {
if (s == null || s.equals("")) {
return null;
}
s = s.replace(" ", "");
byte[] baKeyword = new byte[s.length() / 2];
for (int i = 0; i < baKeyword.length; i++) {
try {
baKeyword[i] = (byte) (0xff & Integer.parseInt(s.substring(i * 2, i * 2 + 2), 16));
} catch (Exception e) {
e.printStackTrace();
}
}
try {
s = new String(baKeyword, "UTF-8");
new String();
} catch (Exception e1) {
e1.printStackTrace();
}
unicode字符转中文return s;
}
*Unicode转为字符串
/**
* unicode 转字符串
*/
public static String unicode2String(String unicode) {
StringBuffer string = new StringBuffer();
String[] hex = unicode.split("\\\\u");
for (int i = 1; i < hex.length; i++) {
// 转换出每⼀个代码点
int data = Integer.parseInt(hex[i], 16);
// 追加成string
string.append((char) data);
}
String();
}
此⽅法虽然解决了转化过程中中⽂乱码的问题,但是过于复杂,笔者后来⼜发现⼀种新的转化⽅式,可直接转化,中⽂不乱码,
代码如下:
*字符串转16进制
/**
* 字符串转换成为16进制(⽆需Unicode编码)
* @param str
* @return
*/
public static String str2HexStr(String str) {
char[] chars = "0123456789ABCDEF".toCharArray();
StringBuilder sb = new StringBuilder("");
byte[] bs = Bytes();
int bit;
for (int i = 0; i < bs.length; i++) {
bit = (bs[i] & 0x0f0) >> 4;
sb.append(chars[bit]);
bit = bs[i] & 0x0f;
sb.append(chars[bit]);
// sb.append(' ');
}
String().trim();
}
*16进制转为字符串
/**
* 16进制直接转换成为字符串(⽆需Unicode解码)
* @param hexStr
* @return
*/
public static String hexStr2Str(String hexStr) {
String str = "0123456789ABCDEF";
char[] hexs = CharArray();
byte[] bytes = new byte[hexStr.length() / 2];
int n;
for (int i = 0; i < bytes.length; i++) {
n = str.indexOf(hexs[2 * i]) * 16;
n += str.indexOf(hexs[2 * i + 1]);
bytes[i] = (byte) (n & 0xff);
}
return new String(bytes);
}
下⾯是补充
java字符串和⼗六进制字符串互转
public class HexStringUtils {
private static final char[] DIGITS_HEX = { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'A', 'B', 'C', 'D',
'E', 'F' };
protected static char[] encodeHex(byte[] data) {
int l = data.length;
char[] out = new char[l << 1];
for (int i = 0, j = 0; i < l; i++) {
out[j++] = DIGITS_HEX[(0xF0 & data[i]) >>> 4];
out[j++] = DIGITS_HEX[0x0F & data[i]];
}
return out;
}
protected static byte[] decodeHex(char[] data) {
int len = data.length;
if ((len & 0x01) != 0) {
throw new RuntimeException("字符个数应该为偶数");
}
byte[] out = new byte[len >> 1];
for (int i = 0, j = 0; j < len; i++) {
int f = toDigit(data[j], j) << 4;
j++;
f |= toDigit(data[j], j);
j++;
out[i] = (byte) (f & 0xFF);
}
return out;
}
protected static int toDigit(char ch, int index) {
int digit = Character.digit(ch, 16);
if (digit == -1) {
throw new RuntimeException("Illegal hexadecimal character " + ch + " at index " + index);
}
return digit;
}
public static String toHex(String str) {
return new String(Bytes()));
}
public static String fromHex(String hex) {
return new String(CharArray()));
}
public static void main(String[] args) {
String s = "abc你好";
String hex = toHex(s);
String decode = fromHex(hex);
System.out.println("原字符串:" + s);
System.out.println("⼗六进制字符串:" + hex);
System.out.println("还原:" + decode);
}
}
toHexString
public static String toHexString(int i)以⼗六进制的⽆符号整数形式返回⼀个整数参数的字符串表⽰形式。
如果参数为负,那么⽆符号整数值为参数加上 232;否则等于该参数。将该值转换为⼗六进制(基数 16)的⽆前导 0 的ASCII 数字字符串。如果⽆符号数的⼤⼩值为零,则⽤⼀个零字符 '0' ('\u0030') 表⽰它;否则,⽆符号数⼤⼩的表⽰形式中的第⼀个字符将不是零字符。⽤以下字符作为⼗六进制数字:
0123456789abcdef
这些字符的范围是从 '\u0030' 到 '\u0039' 和从 '\u0061' 到 '\u0066'。如果希望得到⼤写字母,可以在结果上调⽤
参数:
i - 要转换成字符串的整数。
返回:
⽤⼗六进制(基数 16)参数表⽰的⽆符号整数值的字符串表⽰形式。
// 转化字符串为⼗六进制编码
public static String toHexString(String s)
{
String str="";
for (int i=0;i<s.length();i++)
{
int ch = (int)s.charAt(i);
String s4 = HexString(ch);
str = str + s4;
}
return str;
}
// 转化⼗六进制编码为字符串
public static String toStringHex(String s)
{
byte[] baKeyword = new byte[s.length()/2];
for(int i = 0; i < baKeyword.length; i++)
{
try
{
baKeyword[i] = (byte)(0xff & Integer.parseInt(s.substring(i*2, i*2+2),16)); }
catch(Exception e)
{
e.printStackTrace();
}
}
try
{
s = new String(baKeyword, "utf-8");//UTF-16le:Not
}
catch (Exception e1)
{
e1.printStackTrace();
}
return s;
}
// 转化⼗六进制编码为字符串
public static String toStringHex(String s)
{
byte[] baKeyword = new byte[s.length()/2];
for(int i = 0; i < baKeyword.length; i++)
{
try
{
baKeyword[i] = (byte)(0xff & Integer.parseInt(s.substring(i*2, i*2+2),16)); }
catch(Exception e)
{
e.printStackTrace();
}
}
try
{
s = new String(baKeyword, "utf-8");//UTF-16le:Not
}
catch (Exception e1)
{
e1.printStackTrace();
}
return s;
}
public static void main(String[] args) {
System.out.println(encode("中⽂"));
System.out.println(decode(encode("中⽂")));
}
/*
* 16进制数字字符集
*/
private static String hexString="0123456789ABCDEF";
/*
* 将字符串编码成16进制数字,适⽤于所有字符(包括中⽂)
*/
public static String encode(String str)
{
//根据默认编码获取字节数组
byte[] Bytes();
StringBuilder sb=new StringBuilder(bytes.length*2);
//将字节数组中每个字节拆解成2位16进制整数
for(int i=0;i<bytes.length;i++)
{
sb.append(hexString.charAt((bytes[i]&0xf0)>>4));
sb.append(hexString.charAt((bytes[i]&0x0f)>>0));
}
String();
}
/*
* 将16进制数字解码成字符串,适⽤于所有字符(包括中⽂)
*/
public static String decode(String bytes)
{
ByteArrayOutputStream baos=new ByteArrayOutputStream(bytes.length()/2);
//将每2位16进制整数组装成⼀个字节
for(int i=0;i<bytes.length();i+=2)
baos.write((hexString.indexOf(bytes.charAt(i))<<4 |hexString.indexOf(bytes.charAt(i+1)))); return new ByteArray());
}
第⼆种⽅法:
将指定byte数组以16进制的形式打印到控制台
package com.nantian.iclient.atm.sdb;
public class Util {
public Util() {
}
/**
* 将指定byte数组以16进制的形式打印到控制台
* @param hint String
* @param b byte[]
* @return void
*/
public static void printHexString(String hint, byte[] b) {
System.out.print(hint);
for (int i = 0; i < b.length; i++) {
String hex = HexString(b[i] & 0xFF);
if (hex.length() == 1) {
hex = '0' + hex;
}
System.out.UpperCase() + " ");
}
System.out.println("");
}
/**
*
* @param b byte[]
* @return String
*/
public static String Bytes2HexString(byte[] b) {
String ret = "";
for (int i = 0; i < b.length; i++) {
String hex = HexString(b[i] & 0xFF);
if (hex.length() == 1) {
hex = '0' + hex;
}
ret += UpperCase();
}
return ret;
}
/**
* 将两个ASCII字符合成⼀个字节;
* 如:"EF"--> 0xEF
* @param src0 byte
* @param src1 byte
* @return byte
*/
public static byte uniteBytes(byte src0, byte src1) {
byte _b0 = Byte.decode("0x" + new String(new byte[]{src0})).byteValue();
_b0 = (byte)(_b0 << 4);
byte _b1 = Byte.decode("0x" + new String(new byte[]{src1})).byteValue();
byte ret = (byte)(_b0 ^ _b1);
版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。
发表评论