8. 多态——编译时类型&运行时类型
一、引用变量的两种类型
1. 编译时类型:由声明该变量时使用的类型决定
2. 运行时类型:由实际赋给该变量的对象决定
如果编译时类型和运行时类型不一致,就可能出现多态。
class BaseClass
{
public int val = 6;
public void base()
{
System.out.println("父类的普通方法");
}
public void test()
{
System.out.println("父类的被覆盖的方法");
}
} public class SubClass extends BaseClass
{
public String val = "测试多态";
public void test()
{
System.out.println("子类的覆盖父类的方法");
}
public void sub()
{
System.out.println("子类的普通方法");
} public static void main(String[] args)
{
// 编译时类型为BaseClass,运行时类型为SubClass
BaseClass bc = new SubClass(); // 访问的是父类对象的实例变量,即输出6
System.out.println(bc.val);
// 调用从父类继承到的base()
bc.base();
// 调用当前类的test()
bc.test();
// 因为bc的编译时类型是BaseClass,而该类没有提供sub(),所以下面代码编译时会出错
// bc.sub();
}
}
注:上面程序中定义的引用变量bc,其编译时类型为BaseClass,而运行时类型为SubClass。当调用引用变量bc的test()方法时,实际执行的是SubClass类中覆盖后的test()方法,这就可能出现多态了。此外,虽然引用变量bc实际所引用的对象确实包含sub()方法(例如,可以通过反射来执行该方法),但由于引用变量在编译阶段只能调用其编译时类型所具有的方法,因此编译时无法调用sub()方法。即引用变量只能调用它编译时类型的方法,而不能调用它运行时类型的方法,即使它实际所引用的对象确实包含该方法。
3. 引用变量在编译阶段只能调用其编译时类型所具有的方法,但运行时则执行它运行时类型所具有的方法
二、多态的原理
1. Java允许把一个子类对象直接赋给一个父类引用对象,这也被称为向上转型
2. 当调用这种引用对象的方法时,其方法行为总是表现出子类方法的行为特征,而不是父类方法的行为特征
3. 这就导致:相同类型的变量调用同一个方法时呈现出多种不同的行为特征,这就是多态
三、引用变量的强制类型转换
1. 如果需要让某个引用变量调用它运行时类型的方法,则必须把它强制转换成运行时类型,即将一个引用类型变量强制转换成其子类类型
- 引用类型之间的转换只能在具有继承关系的两个类型之间进行,否则编译时就会出现错误
- 如果试图把一个父类实例转换成子类类型,则这个对象必须实际上是子类实例才行(即其运行时类型是子类类型)
public class ConversionTest
{
public static void main(String[] args)
{
double d = 3.14;
long l = (long)d; // objStr变量的编译时类型为Object,运行时类型为String
Object objStr = "Hello";
// 由于Object与String存在继承关系,可以强制类型转换
String str = (String)objStr; // objInt变量的编译时类型为Object,运行时类型为Integer
Object objInt = Integer.valueOf(5);
// 由于Object与Integer存在继承关系,可以强制类型转换
Integer intNum = (Integer)objInt;
// 虽然Object与String存在继承关系,可以强制类型转换
// 但是objInt的运行时类型不是String,故下面代码运行时引发ClassCastException异常
// String str = (String)objInt;
}
}
2. 由于进行强制类型转换时可能出现异常,因此进行类型转换之前应先通过instanceof运算符来判断是否可以成功转换,使程序更健壮
- 通常先用instanceof判断一个对象是否可以强制类型转换,然后再使用(type)运算符进行强制类型转换,从而保证程序不会出现错误
if(objInt instanceof String)
{
String str = (String)objInt;
}
四、instanceof运算符
1. instanceof是Java提供的运算符,与+、-等算术运算符的用法大致相似
2. 用法:实例 instanceof 类名(或接口名)
- 实例的编译时类型要么与后面的类相同,要么与后面的类具有父子继承关系,否则会引起编译错误
3. 作用:判断其左侧的实例是否是右侧的类(或者其子类、实现类)的实例
- 在进行instanceof运算时,左侧的实例以其运行时类型为判断依据
public class InstanceofTest
{
public static void main(String[] args)
{
Object objStr = "Hello"; // objStr的编译时类型是Object,故可进行instanceof运算
if(objStr instanceof Object) // 返回true
{
System.out.println("字符串是Object类的实例");
} // objStr的编译时类型是Object,Object类与String类存在继承关系
// 所以可以进行instanceof运算
if(objStr instanceof String) // 返回true
{
System.out.println("字符串是String类的实例");
} // objStr的编译时类型是Object,Object类与Math类存在继承关系
// 所以可以进行instanceof运算
if(objStr instanceof Math) // 返回false
{
}
else
{
System.out.println("字符串不是Math类的实例");
} String str = "Hello";
// String类与Math类没有继承关系,所以下面的代码无法编译通过
// if(str instanceof Math) { ... }
}
}
8. 多态——编译时类型&运行时类型的更多相关文章
- 正确理解java编译时,运行时以及构建时这三个概念
Java中的许多对象(一般都是具有父子类关系的父类对象)在运行时都会出现两种类型:编译时类型和运行时类型,例如:Person person = new Student();这行代码将会生成一个pers ...
- 深入浅出OOP(三): 多态和继承(动态绑定/运行时多态)
在前面的文章中,我们介绍了编译期多态.params关键字.实例化.base关键字等.本节我们来关注另外一种多态:运行时多态, 运行时多态也叫迟绑定. 运行时多态或迟绑定.动态绑定 在C#语音中,运行时 ...
- Java源文件编译成功但是运行时加载不到文件
最近系统重装了一些,Java等环境变量都需要重新配置,配置好以后编写了一个Java源文件编译了一下,通过Javac编译源文件,编译成功,但是再通过Java运行时没找到报出找不到加载文件或者加载文件不存 ...
- Linux下显示运行时链接(运行时加载)
目录 介绍 如何加载动态库 dlopen() 第一个参数: 被加载动态库的路径 第二个参数: flag表示函数符号的解析方式 dlopen 返回值 dlsym() 参数: 返回值 符号优先级 dler ...
- 在引用KindEditor编辑器时,运行时出现以下错误:错误46 找不到类型或命名空间名称“LitJson”(是否缺少 using 指令或程序集引用?)
将asp.net下bin文件夹下的文件LitJSON.dll拷贝到工程的bin目录下,并在工程中添加引用 在后台加入: using LitJson;
- gcc新版本号引起的编译错误(命令运行时的外部库输入位置)
昨天,遇到一个比較bug的错误,用gcc来编译几个简单的文件出错,编译环境为x86_64的Ubuntu12.04.gcc版本号号例如以下: gcc (Ubuntu/Linaro 4.6.3-1ubun ...
- 使用Spring4时, 运行时出现找不到MappingJacksonHttpMessageConverter的情况
启动项目报错: [org.springframework.web.context.ContextLoader]Context initialization failed org.springframe ...
- Spring3升级到Spring4时, 运行时出现找不到MappingJacksonHttpMessageConverter的情况
[org.springframework.web.context.ContextLoader]Context initialization failed org.springframework.bea ...
- [调试]VS2013调试时提示“运行时当前拒绝计算表达式的值”
VS2013 下单元测试调试时遇到的问题,以前倒从未遇到过. 中文关键字在百度和谷歌中搜索均无果. Google 下搜索 “The runtime has refused to evaluate th ...
随机推荐
- 中文版 Apple 官方 Swift 教程《The Swift Programming Language》
简介 欢迎使用 Swift 关于 Swift 版本兼容性 Swift 初见 Swift 版本历史记录 Swift 教程 基础部分 基本运算符 字符串和字符 集合类型 控制流 函数 闭包 枚举 类和结构 ...
- 面向对象OPP
在此之前学习的编程方式均称为面向过程,过程类似于函数,只能执行,没有返回值 面向过程和面向对象 面向过程-->怎么做? 面向对象-->谁来做? 相比函数,面向对象 是更大的封装,根据职 ...
- go语言设计模式之state
state.go package main import ( "fmt" "math/rand" "os" "time" ...
- Codeforces Round #608 (Div. 2)
传送门 A. Suits 签到. Code /* * Author: heyuhhh * Created Time: 2019/12/15 17:16:33 */ #include <iostr ...
- Go 字符串 (string)
字符串类型为 string,使用双引号或者反引号包起来 字符串形式 反引号 当使用反引号时不会对字符串进行转义,并可以包含多行文本 示例: package main import "fmt& ...
- 设计模式-Composite(结构型模式) 用于 递归构建 树 状 的组合结构,与Decorator的区别是 Composite旨在通过构造子类而添加新操作,而Decorator直接添加新操作。
以下代码来源: 设计模式精解-GoF 23种设计模式解析附C++实现源码 //Component.h #pragma once class Component { public: Component( ...
- 2019徐州网络赛 I J M
I. query 比赛时候没有预处理因子疯狂t,其实预处理出来因子是\(O(nlog(n))\)级别的 每个数和他的因子是一对偏序关系,因此询问转化为(l,r)区间每个数的因子在区间(l,r)的个数 ...
- BERT-wwm、BERT-wwm-ext、RoBERTa、SpanBERT、ERNIE2
一.BERT-wwm wwm是Whole Word Masking(对全词进行Mask),它相比于Bert的改进是用Mask标签替换一个完整的词而不是子词,中文和英文不同,英文中最小的Token就是一 ...
- 前端 web mime类型引起的 常见404错误
mime 类型设置参考https://www.w3school.com.cn/media/media_mimeref.asp
- React: React组件创建的三种方式
一.简介 在前面介绍的React组件知识中,对于组件的创建我只是用了其中某一种方式.其实,在2013年React诞生之初,对于React组件的创建,仅仅只有一种方式,也即createClass函数,在 ...