JAVA引用变量有两个类型:一个是编译时类型,一个运行时类型
编译时类型由声明该变量时使用的类型决定,运行时类型由实际赋给该变量的对象决定。如果编译时类型和运行时类型不一样,就形成了多态。
因为子类其实是一个特殊的父类,因为JAVA允许把一个子类对象直接赋给一个父类引用变量,无须任何类型转换,称为向上转型,由系统自动完成。
当把一个子类对象直接赋给父类引用变量后,当运行时调用该引用变量的方法时,其方法行为总是表现出子类方法的行为特征,而不是父类方法的行为特征,这就可能出现:相同类型的变量、调用同一个方法时呈现出多种不同的行为特征,这就是多态
也就是说,当子类对象赋给父类引用变量时候,就发生了多态。在调用的时候,属性还是直接调用父类的属性,如果子类重写了父类的方法,则调用子类的方法,否则继续调用父类方法,不能调用子类独有的方法
强制类型转换
编写JAVA程序时,引用变量只能调用它编译时类型的方法,而不能调用它运行时类型的方法,即使它所引用的对象确实包含该方法。如果需要让这个引用变量调用它运行时类型的方法,则必须把它强制类型转换成运行时类型,强制类型转换需要借助于类型转换运算符。
类型转换运算符是小括号。
基本类型之间的转换只能在数值类型之间进行,这里所说的数值类型包括整数型、字符型、和浮点型。但数值类型和布尔类型之间不能进行转换。
引用类型之间的转换只能在具有继承关系的两个类型之间进行,如果是两个没有任何继承关系的类型,则无法进行类型转换,否则编译时就会出现错误。如果视图把一个父类实例转换成子类实例,则这个对象必须实际上是子类实例才行,否则会出现ClassCastException异常
使用instanceof运算符可以让强制类型转换更安全
instanceof关键字
instanceof运算符的前一个操作数通常是一个引用类型变量,后一个操作数通常是一个类(也可以是一个接口),他用于判断前面的对象是否是后面的类,或者其子类、实现类的实例,如果是,则返回true,否则返回false
**如果子类重写了父类的方法,那么父类的构造函数中如果调用了被子类重写的方法,则会产生空指针