P55: 如果要执行窄化转换(
narrowing conversion
)的操作(也就是说,将能够容纳更多信息的数据类型转换成无法容纳那么多信息的类型),就有可能面临信息丢失的危险。此时,编译器强制要求我们进行类型转换,这实际上是说:“这可能是一件危险的事情,如果无论如何都要这么做,必须显式进行类型转换。”而对于扩展转换(widening conversion
),则不必显式的进行类型转换,因为新类型肯定能够容纳原来类型的信息,不会造成任何信息的丢失。
上面这段话应该分成基本数据类型和引用类型两个方面进行讨论。
基本数据类型
int a = 1;
float b = 91.1f;
//需要进行强制类型转换,转换后的a的值是91,剩下的0.1则丢失了。
a = (int) b;
//不需要进行强制类型转换
b = a;
如上述代码所示,b
的类型是float
,拥有比int
更多的内容,转成int
类型会损失部分数据,所以我们需要使用(int)
进行强制转换。而从int转成float则不需要进行强制转换,因为转换过程中不会丢失任何数据。
引用类型
class TestA {
int a = 11;
}
class TestB extends TestA {
int b = 1;
}
class TestC {
int a = 11;
}
public class Test {
public static void main(String[] args) {
//不需要强制类型转换
TestA a = new TestB();
//需要强制类型转换
TestB b = (TestB) a;
//会抛出java.lang.ClassCastException
TestB bb = (TestB) new TestA();
}
}
与基本数据类型不同的是,我们的对象是存在堆上面的,不管怎么转换,其内容都是不变的,改变的只是该对象所能够展示的能力,也就是其能够调用的方法和变量。
首先TestB
是继承TestA
的,TestB
对象就是一个TestA
对象。这里是和基本数据类型不一样的,因为它在堆上的对象是不会发生改变的,也就不存在数据的丢失问题。所以下面语句不需要强制转换,也不会报异常。
TestA a = new TestB();
然后我们想把TestB
对象重新被TestB
引用,那就必须使用强制类型转换了。
//需要强制类型转换
TestB b = (TestB) a;
因为现在编译器只知道a
是一个TestA
对象,TestA
有TestB
和TestC
两个子类。编译器要求强转是提醒我们注意这部分可能出错。如果抛开编译器看的话,不强转运行是不会出错的。
//会抛出java.lang.ClassCastException
TestB bb = (TestB) new TestA();
现在我们看最后一部分,这行代码在编译的时候是不会报错的,因为TestA
与TestB
存在继承关系,所以编译能够通过。但是在运行的时候,TestA
对象是不能够转成TestB
对象的,因为其在堆上是不发生变化的,TestB
的一些独有的变量和方法,TestA
对象是不能够显示出来的。如果我们在转换的时候不报错,那么使用bb
来调用一些TestB
独有的方法的时候一样还是要报错的。