美文网首页Java 杂谈
Think in Java学习笔记 第三章 操作符

Think in Java学习笔记 第三章 操作符

作者: Rreply | 来源:发表于2019-03-16 23:28 被阅读1次

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对象,TestATestBTestC两个子类。编译器要求强转是提醒我们注意这部分可能出错。如果抛开编译器看的话,不强转运行是不会出错的。

//会抛出java.lang.ClassCastException
TestB bb = (TestB) new TestA();

现在我们看最后一部分,这行代码在编译的时候是不会报错的,因为TestATestB存在继承关系,所以编译能够通过。但是在运行的时候,TestA对象是不能够转成TestB对象的,因为其在堆上是不发生变化的,TestB的一些独有的变量和方法,TestA对象是不能够显示出来的。如果我们在转换的时候不报错,那么使用bb来调用一些TestB独有的方法的时候一样还是要报错的。

相关文章

网友评论

    本文标题:Think in Java学习笔记 第三章 操作符

    本文链接:https://www.haomeiwen.com/subject/yzzsmqtx.html