事件起因

在做一个类型转换时遇到了下面的异常:

java.lang.ClassCastException - 图1

是说不能把 Integer 类型强制转换成 String 类型。

我的测试代码大概是这个样子:

  1. public static void testClassCastException() {
  2. Integer i1 = Integer.valueOf(getT()); // ① 抛异常
  3. Integer i2 = (Integer) getT(); // ② 正常
  4. Integer i3 = Integer.valueOf(getT().toString()); // ③ 正常
  5. }
  6. private static <T> T getT() {
  7. return (T) new Integer(10);
  8. }
  9. // ①和③的不同在于Integer.valueOf()里的参数的类型,其实Integer.valueOf()方法就是需要String类型的,所以我们直接用Integer类型当然在底层是不识别的。
  10. // ①和②的不同在于类型转换的方式,这个就引申出下面这个问题了。

(String)、toString、String.valueOf 的区别

  1. toString()
    toString()是在 Object 中定义的,因此,任何继承 Object 的类都具有这个方法。

在 API 中 toString()被描述为:

返回该对象的字符串表示。通常,toString 方法会返回一个“以文本方式表示”此对象的字符串。结果应是一个简明但易于读懂的信息表达式。建议所有子类都重写此方法。

Object 类的 toString 方法返回一个字符串,该字符串由类名(对象是该类的一个实例)、标记符“@”和此对象哈希码的无符号十六进制表示组成。换句话说,该方法返回一个字符串,它的值等于:getClass().getName()+’@’+Integer.toHexString(hashCode())
  我们在定义一个类的时候可以重写继承自 Object 的 toString()方法,以此来表示该对象的基本信息。
  但是,使用 toString()的对象不能为 null,否则会抛出异常 java.lang.NullPointerException。

  1. String.valueOf()
    String.valueOf()解决了 toString()使用对象不能为空的问题,实际上,该方法在底层还是使用了 toString()。如下源码:
  1. public static String valueOf(Object obj) {
  2. return (obj == null) ? "null" : obj.toString();
  3. }

值得注意的是,如果使用对象为 null,返回的结果是字符串”null”,而不是对象 null。
  在很多时候使用 String.valueOf()比使用 toString()更能减少报错的机会,因为不用考虑是否为 null 的情况。但是,也要特别注意:if(String.valueOf(o) == null){…}这样的代码肯定就回有问题,因为永远不会得到执行。

  1. (String)
    (String)区别于上面两种方法,因为它是强制转换。
    每个对象在创建的时候就已经确定了类型不能改变,所谓的强制转换只是表面上将其转换成了另一种类型,就相当于 A 被当成了 B 使用,如果 A 能够被当成 B,那一切没有问题,但是 A 做不了 B,就只能抛出异常,说我做不到。
  1. Integer o = new Integer(100);
  2. System.out.println((String)o);

如上代码编译时就会报错:Cannot cast Integer to String,说明 Integer 不能通过强制转换成 String。
  而我们一开始的代码:

  1. Object x = new Integer(0);
  2. System.out.println((String)x);

在编译时没错,运行时抛出异常。这是因为 x 在表面上是 Object,实际上是 Integer。而 Object 是可以通过强制转换成为 String 的。所以,在编译的时候,x 被当成了 Object,大家相安无事,真正运行的时候,x 被查出来是 Integer,理所当然就抛出了 ClassCastException。

最安全的方法:instanceof

取到的对象进行类型转换如果不知道具体会是什么类型的,可以通过instanceof关键字来检查改对象属于哪个类型的,再具体去进行某些业务。

PS:a instanceof b的结果和b.class.isInstance(a)的结果是一致的,可以实现相同的功能,完全等价。

为什么在 Java 里不能将 Integer 强制转换成 String

因为 StringInteger 不是在同一个对象阶层。

  1. Object
  2. / \
  3. / \
  4. String Integer

当你尝试强制转换时,仅仅会在同一个对象阶层转换。比如:

  1. Object
  2. /
  3. /
  4. A
  5. /
  6. /
  7. B

在这种情况,(A)objB 或者 (Object)objB 或者 (Object)objA 可以进行转换。

正如其他人已经提到,将integer转换成String可以使用以下方法:

基本类型的整型时使用:String.valueOf(integer)或者Integer.toString(integer) 来转换成String
Integer对象型时使用:Integer.toString()转换成String

参考:

翻译|为什么在 Java 里不能将 Integer 强制转换成 String | PostgreSQL DBA

java.toString() ,(String),String.valueOf 的区别 _java_springk 的专栏-CSDN 博客


求关注、分享、在看!!!
你的支持是我创作最大的动力。

java.lang.ClassCastException - 图2