今日看公司代码时发现,在string和byte数组转换的过程中,大量的无聊try catch。所以写了本文,作一个java基本编程知识的小科普。
分享一个java编程的小技巧,简单实用。
建议
其实内容就一句话:
在做String和byte[]的相互转换时,请使用StandardCharsets.UTF_8来替代”utf-8”
解释一下,通常我们代码是这样写:
String string = new String(bytes, "utf-8");
byte[] bytes = string.getBytes("utf-8");
请换成下面这个写法:
String string = new String(bytes, StandardCharsets.UTF_8);
byte[] bytes = string.getBytes(StandardCharsets.UTF_8);
注: 其实大家看到这里就可以打住了
问题解释
第一个写法功能当然没问题,但是代码写完之后,IDE一定会立刻提醒你,这里会抛出UnsupportedEncodingException。
而最要命的是,UnsupportedEncodingException是继承自类Exceptio。
这是一个checked exception, 这是一个checked exception, 这是一个checked exception!
JDK的源代码如下:
public String(byte bytes[], String charsetName)
throws UnsupportedEncodingException {
this(bytes, 0, bytes.length, charsetName);
}
这意味着我们要不加上try catch,要不就要在方法上显示申明要抛出异常。
而申明抛出UnsupportedEncodingException异常绝对不是一个好注意,鬼都知道这里这个UTF-8一定不会
unsupported,把这个麻烦扔给调用者绝对是一个不负责任的行为。
所以我在代码中看到大量的类似代码:
try {
String json = new String(data, "utf-8");
// 此处略去××字
} catch (UnsupportedEncodingException e) {
e.printStackTrace();
}
解决方法
这个问题由来已久,java社区解决这个问题的方式也很早就有,没有记错的话 apache commons大概十年前就提供了
方案,注意jdk中提供的另外一个不抛UnsupportedEncodingException的构造函数:
public String(byte bytes[], Charset charset) {
this(bytes, 0, bytes.length, charset);
}
和前一个构造函数的差别就是这里直接输入了Charset对象,不需要做一次从string到Charset
的转化(这里才是UnsupportedEncodingException抛出的根源)。而我们日常要用到的charset是非常
有限的,因此只要简单列常来最常用的几个就好了。
JDK7之后,java引入了java.nio.charset.StandardCharsets来做charset预定义:
public final class StandardCharsets {
public static final Charset US_ASCII = Charset.forName("US-ASCII");
public static final Charset ISO_8859_1 = Charset.forName("ISO-8859-1");
public static final Charset UTF_8 = Charset.forName("UTF-8");
......
}
在jdk之前,很多基本类库都提供类似的功能,比如大家熟悉的apache commons,这个是最早提供也是用的最多的的:
org.apache.commons.codec.Charsets
/**
* @see Standard charsets
* @deprecated Use Java 7's {@link java.nio.charset.StandardCharsets}
*/
@Deprecated
public static final Charset UTF_8 = Charset.forName(CharEncoding.UTF_8);
注意上面的注释,现在apache已经Deprecated 它了,建议大家用StandardCharsets。