JAVA 十六进制与字符串的转换

 更新时间:2009年05月14日 23:42:38   投稿:mdxy-dxy  
笔者前几日在开服过程中需要将字符串转化成为16进制的字符串,在网上找到了一些方法尝试之后,均发现存在一个问题-->字符串转为16进制后再转回来,英文正常,中文出现乱码

笔者前几日在开服过程中需要将字符串转化成为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" + Integer.toHexString(c));
  }
  return unicode.toString();
}

*字符串转为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 = Integer.toHexString(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();
 }
 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);
 }
 return string.toString();
}

此方法虽然解决了转化过程中中文乱码的问题,但是过于复杂,笔者后来又发现一种新的转化方式,可直接转化,中文不乱码,

代码如下:

*字符串转16进制

/**
 * 字符串转换成为16进制(无需Unicode编码)
 * @param str
 * @return
 */
public static String str2HexStr(String str) {
 char[] chars = "0123456789ABCDEF".toCharArray();
 StringBuilder sb = new StringBuilder("");
 byte[] bs = str.getBytes();
 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(' ');
 }
 return sb.toString().trim();
}

*16进制转为字符串

/**
 * 16进制直接转换成为字符串(无需Unicode解码)
 * @param hexStr
 * @return
 */
public static String hexStr2Str(String hexStr) {
 String str = "0123456789ABCDEF";
 char[] hexs = hexStr.toCharArray();
 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(encodeHex(str.getBytes()));
 }

 public static String fromHex(String hex) {
  return new String(decodeHex(hex.toCharArray()));
 }

 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'。如果希望得到大写字母,可以在结果上调用 String.toUpperCase() 方法:

Integer.toHexString(n).toUpperCase()

参数:
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 = Integer.toHexString(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=str.getBytes(); 
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)); 
} 
return sb.toString(); 
} 
/* 
* 将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 String(baos.toByteArray()); 
} 

第二种方法:

将指定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 = Integer.toHexString(b[i] & 0xFF); 
if (hex.length() == 1) { 
hex = '0' + hex; 
} 
System.out.print(hex.toUpperCase() + " "); 
} 
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 = Integer.toHexString(b[i] & 0xFF); 
if (hex.length() == 1) { 
hex = '0' + hex; 
} 
ret += hex.toUpperCase(); 
} 
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); 
return ret; 
} 

/** 
* 将指定字符串src,以每两个字符分割转换为16进制形式 
* 如:"2B44EFD9" --> byte[]{0x2B, 0x44, 0xEF, 0xD9} 
* @param src String 
* @return byte[] 
*/ 
public static byte[] HexString2Bytes(String src){ 
byte[] ret = new byte[8]; 
byte[] tmp = src.getBytes(); 
for(int i=0; i<8; i++){ 
ret[i] = uniteBytes(tmp[i*2], tmp[i*2+1]); 
} 
return ret; 
} 

} 

以上就是JAVA 十六进制与字符串的转换的详细内容,更多关于JAVA 十六进制的资料请关注脚本之家其它相关文章!

相关文章

  • 浅谈java中Map的用法

    浅谈java中Map的用法

    Map简介:将键映射到值的对象。一个映射不能包含重复的键;每个键最多只能映射到一个值。此接口取代 Dictionary 类,后者完全是一个抽象类,而不是一个接口。
    2015-09-09
  • Java Netty核心模块超详细梳理

    Java Netty核心模块超详细梳理

    Netty是一个java开源项目,是一个异步的、基于事件驱动的网络应用框架,用以开发高性能、高可用的网络io程序,这篇文章主要介绍了Netty核心模块
    2022-11-11
  • java发送HttpClient请求及接收请求结果过程的简单实例

    java发送HttpClient请求及接收请求结果过程的简单实例

    下面小编就为大家带来一篇java发送HttpClient请求及接收请求结果过程的简单实例。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2016-11-11
  • java 中的volatile关键字

    java 中的volatile关键字

    这篇文章主要介绍了java 中的volatile关键字,volatile在多处理器开发中保证共享变量的“可见性”。可见性的意思是当一个线程修改一个共享变量时,另一个一个线程立马可以读到这个修改的值。下面我们来看看文章的具体介绍内容吧

    2021-12-12
  • Java设计模式之单例模式简单解析

    Java设计模式之单例模式简单解析

    这篇文章主要介绍了Java设计模式之单例模式简单解析,单例模式的优点在于在内存中某个类只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例,避免对资源的多重暂用,需要的朋友可以参考下
    2023-12-12
  • Redisson公平锁的源码解读分享

    Redisson公平锁的源码解读分享

    Redisson是一个在Redis的基础上实现的Java驻内存数据网格(In-Memory Data Grid)。这篇文章主要通过源码和大家聊聊Redisson公平锁,需要的可以了解一下
    2022-11-11
  • Java中从键盘输入多个整数的方法

    Java中从键盘输入多个整数的方法

    今天小编就为大家分享一篇Java中从键盘输入多个整数的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-07-07
  • java客户端线上Apollo服务端的实现

    java客户端线上Apollo服务端的实现

    这篇文章主要介绍了java客户端线上Apollo服务端的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2019-08-08
  • Java JDK与cglib动态代理有什么区别

    Java JDK与cglib动态代理有什么区别

    这篇文章主要介绍了Java JDK动态代理和cglib动态代理的区别文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习吧
    2023-03-03
  • RestFul风格 — 使用@PathVariable传递参数报错404的解决

    RestFul风格 — 使用@PathVariable传递参数报错404的解决

    这篇文章主要介绍了RestFul风格 — 使用@PathVariable传递参数报错404的解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10

最新评论