1.从对象的内存角度来理解.
假设现在有一个父类Father,它里面的变量需要占用1M内存.有一个它的子类Son,里面的变量需要占用0.5M内存.
现在通过代码来看看内存的分配情况:
2.f = new Father();//系统将分配1M内存.
Son s = new Son();//系统将分配1.5M内存!因为子类中有一个隐藏的引用super会指向父类实例,所以在实例化子类之前会先实例化一个父类,也就是说会先执行父类的构造函数.由于s中包含了父类的实例,所以s可以调用父类的方法.
3.Son s1 = s;//s1指向那1.5M的内存.
Father f1 = (Father)s;//这时f1会指向那1.5M内存中的1M内存,即是说,f1只是指向了s中实例的父类实例对象,所以f1只能调用父类的方法(存储在1M内存中),而不能调用子类的方法(存储在0.5M内存中).
Son s2 = (Son)f;//这句代码运行时会报ClassCastException.因为f中只有1M内存,而子类的引用都必须要有1.5M的内存,所以无法转换.
Son s3 = (Son)f1;//这句可以通过运行,这时s3指向那1.5M的内存.由于f1是由s转换过来的,所以它是有1.5M的内存的,只是它指向的只有1M内存.
然后是我自己读完的想法:
1, 这句话挺重要的:在子类的构造函数中,子类会隐藏的引用super指向父类实例,所以在实例化子类之前,会先执行父类的构造函数。子类对象包含了父类的实例。
2, 在强制转化时,子类可以向父类转化,因为可以忽略一些,然而父类却不能向子类转化,你不能一句强制转化就让它凭空增加一些内存。通俗的说是一个父亲, 永远不可能去装儿子的(只有我们的父类对象本身就是用子类new出来的时候, 才可以在将来被强制转换为子类对象.)
Father father= new Son();
Son son = (Son) father;