编译类型,运行类型,多态,强转,向上转型,instanceof
具体见《java讲义》p149-151
关于强类型转换:
1. 基础类型间可转
2. 对象间强转, 只能由编译类型转到运行时类型,(即父类转子类,且子类是运行时类型)
多态定义:把子类对象直接赋给父类的引用时,当运行时调用该引用类型的方法时,其方法行为总是表现出子类的行为特征,而不是父类的行为特征,这就出现了同一引用调用引用变量带的方法是,表现出不同的特征,这就是多态。
我一直都对这个编译时类型和运行时类型很迷惑,百度了很多讲的都很浅显大都如下:
Java的引用变量有两个类型,一个是编译时类型,一个是运行时类型,编译时类型由声明该变量时使用的类型决定,运行时类型由实际赋给该变量的对象决定。
这样的说法只是简单地描述了编译时类型、运行时类型,并没有真正的从本质上说明什么是编译时类型什么是运行时类型。最终我还是找到了一篇写的比较好的文章,拷贝其中的一段给大家看看,外加上自己的代码
Person p=new Women()(Women类继承自Person类)那么,假如p的属性修饰符为public 访问属性时得到的是Person类的属性还是Women类的属性,方法调用又是哪个类?答案:会得到Person类的属性,调用Women类的方法。为什么会这样呢?这里就需要知道什么是编译时类型和运行时类型,Java程序状态会分为编译和运行这两种状态,编译时,JVM会在栈中静态创建基本数据变量,和引用数据变量的引用,回到刚刚那句代码,显然,p这个引用就是在编译时创建的,那么,p的编译时类型就是Person了,当运行这句java代码时,JVM在堆中为p新建一块内存,对应new Women()这句代码,所以p的运行时类型就是Women
这篇文章的地址在这大家可以去看看原版文章写得非常好。http://blog.csdn.net/qq_23419401/article/details/52064871#java
下面我就要发表一下自己的看法了。我们都知道栈和堆,在栈中存放的是一些引用和一些变量,而堆内存中存放的是对象。
所以:编译时期可以理解为堆内存中还没有为该对象创建内存,只是在栈中创建了一些基本类型的变量和引用,所以编译时类型就是指在new该对象之前被加载到栈中的属性或方法。而运行时类型,是指new了一个对象出来,在栈中为该对象分配了内存,此时的运行时类型也就是栈中对象的属性或方法了。
Person person = new student();
这里就可以很好地理解编译时类型和运行时类型了。
编译时类型是Person,当new了一个对象之后,堆内存中产生了一个Student对象,此时的类型就是运行时类型,也就是Student类型了。
这里需要补充的是,在多态中,将子类对象赋给父类的引用,会产生多态。
Father son = new Son();
而这里的son只能调用Father的方法(重写或者继承的方法),而不能调用本身定义的一些方法(自己编写的父类中没有的方法)。例如下面这样就会报错。
public class Father {
public String name;
public Father(){
this.name = "father";
}
public void show(){
System.out.println("father's name : " + this.name);
}
}
public class Son extends Father {
public String name;
public Son(){
this.name = "son";
}
public void show(){
System.out.println("Son's name: " + this.name);
}
public void show2(){
System.out.println("show2");
}
public static void main(String[] args) {
Father son = new Son();
son.show2();//此处就会报错
}
}
因为多态中强调:编写java程序时,引用类型变量只能调用其编译时类型的变量,不能调用其运行时类型变量。此时的show2()就是运行时类型的变量了。所以会报错