instanceof 对象类型转换

instanceof 关键字的使用

编译时类型(声明类型)与运行时类型(实例类型)

  1. 编译时类型(声明类型)

    • 编译时类型是指变量声明时所指定的类型,或者说是变量的静态类型。
    • 这个类型在代码编译时就已经确定,编译器根据这个类型来进行类型检查和类型推断。
    • 编译时类型决定了该变量可以调用哪些方法和访问哪些成员变量。
  2. 运行时类型(实例类型)

    • 运行时类型是指对象在内存中实际的类型,或者说对象的实际类。
    • 这个类型是在程序运行时动态确定的,可以通过 instanceof 进行检查。
    • 运行时类型决定了该对象真正所拥有的属性和方法。
    Person o3 = new Student();//那这个是什么类型
    

    Person o3 = new Student(); 这一行代码涉及的是 多态 的概念,具体来说,它是父类引用指向子类对象。

    解释:

    • 声明类型Persono3 的声明类型,意味着编译时 o3 被视为 Person 类型。
    • 实例化类型new Student() 创建了一个 Student 类型的对象。虽然 o3Person 类型的引用,但它指向的是 Student 类型的对象。

    关键点:

    • 声明类型(编译时类型)o3 的声明类型是 Person。这意味着 o3 只能调用 Person 类中定义的方法(包括父类 Person 中继承过来的方法),不能直接调用 Student 类中特有的方法(除非强制类型转换)。
    • 实例类型(运行时类型)o3 实际上引用的是 new Student() 创建的对象,new Student() 创建的对象的实际类型是 Student。这意味着在运行时,o3 指向的对象是 Student 类型,而 StudentPerson 的子类,因此可以通过 instanceof 进行判断。

instanceof 的工作原理

instanceof 是一个用于判断对象是否是某个类的实例,或者是否是该类子类的实例的运算符。其工作原理依赖于 编译时类型运行时类型

instanceof 判断规则:

  • 左侧对象(变量)的类型:左侧对象的 编译时类型运行时类型 都会影响 instanceof 判断。如果编译时类型与目标类类型不兼容,编译时就会报错。
  • 右侧类(目标类型)的类型:如果左侧对象的 运行时类型 是目标类或目标类的子类,则返回 true,否则返回 false

总结

  • 编译时类型:是变量声明时所给定的类型,决定了变量可以调用哪些方法,进行哪些操作。
  • 运行时类型:是对象在内存中实际的类型,决定了对象可以访问哪些成员。
  • instanceof 的工作原理
    • 在编译时检查左边的类型与右边的类是否兼容。
    • 在运行时检查左边对象的实际类型(运行时类型)是否是右边类的实例或子类实例。

CAL 总结一句话!:instanceof通过不通过看声明类型(编译时类型)是否兼容,返回值看实例类型(运行时类型)是不是calssname 的实例或者子类的实例

父类 Person

package com.oop.demo06;public class Person {public Person() {}
}

子类 Student

package com.oop.demo06;public class Student extends Person{public Student() {}
}

子类Teacher

package com.oop.demo06;public class Teacher extends Person {String name;public Teacher() {}public Teacher(String name) {this.name = name;}
}

启动类Application

package com.oop.demo06;
//此例主要讨论 instanceof 关键字的使用
/*
* 声明时类型(编译时类型) 左边的
* 实例类型(运行时类型)  右边的
* 使用格式:object instanceof ClassName
* 所以说 instanceof 编译通过不通过 是看声明类型(编译时类型)是否兼容(是彼此,或者直接继承就是彼此)
* 返回值是true还是false 是看实例类型(运行时类型)是不是 calssname 的实例或者子类的实例
* 是就返回ture 不是就返回false
* */
import sun.rmi.transport.ObjectTable;public class Application {public static void main(String[] args) {// Object >  Person > Student// Object > Person  > Teacher// Object > String   Ctrl +H 打开结构树Student o1 = new Student();System.out.println(o1 instanceof Student);  // true//System.out.println(o1 instanceof Teacher);  // 编译错误,o1 是 Student 类型,不能转换为 Teacher//System.out.println(o1 instanceof String);  // 编译错误,o1 是 Student 类型,不能转换为 StringSystem.out.println(o1 instanceof Person);  // trueSystem.out.println(o1 instanceof Object);  // trueSystem.out.println("============");//Teacher o2 = new Student();编译不通过Teacher o2 = new Teacher();//System.out.println(o2 instanceof Student);  // 编译错误,o2 是 Teacher 类型,不能转换为 Student//System.out.println(o2 instanceof String);  // 编译错误,o2 是 Teacher 类型,不能转换为 StringSystem.out.println(o2 instanceof Teacher);  // trueSystem.out.println(o2 instanceof Person);  // trueSystem.out.println(o2 instanceof Object);  // trueSystem.out.println("============");Person o3 = new Student();//System.out.println(o3 instanceof String);  // 编译错误,o3 编译时是 Person 类型,不能转换为 StringSystem.out.println(o3 instanceof Student);  // true  o3运行时时 Student()的实例 可由Student实例化得到System.out.println(o3 instanceof Teacher);  // falseSystem.out.println(o3 instanceof Person);  // true o3运行时时 Student()的实例 可由Person子类Student实例化得到System.out.println(o3 instanceof Object);  // trueo3运行时时 Student()的实例 可由Obeject的子(Person)->子(Student)类实例化得到System.out.println("============");Person o4 = new Teacher();//System.out.println(o4 instanceof String);  // 编译错误,o4 编译是是 Person 类型,不能转换为 StringSystem.out.println(o4 instanceof Student);  // falseSystem.out.println(o4 instanceof Teacher);  // trueSystem.out.println(o4 instanceof Person);  // trueSystem.out.println(o4 instanceof Object);  // trueSystem.out.println("============");Object o5 = new Student();System.out.println(o5 instanceof String);  // false    o5 是obeject类型System.out.println(o5 instanceof Student);  // trueSystem.out.println(o5 instanceof Teacher);  // falseSystem.out.println(o5 instanceof Person);  // true}
}

对象类型转换

1. 小转大:子类转换为父类(向上转型)

这是允许的,因为每个子类实例都可以视为父类的实例,子类自动符合父类的结构。所以子类对象可以赋值给父类引用,或者直接转换成父类类型。

// 小转大:子类转父类
Son s1 = new Son();
Father f1 = s1;  // 隐式转换(向上转型),这是合法的
f1.run();  // 调用父类的run方法// 或者强制转换
((Father)s1).run();  // 显式转换,调用父类的run方法

解释:

  • 子类对象可以赋值给父类引用:Father f1 = s1; 这样可以直接用子类对象的父类部分。
  • 强制转换 ((Father)s1).run(); 也是合法的,因为 s1 实际上指向的是 Son 类型的对象。

2. 大转小:父类转换为子类(向下转型)

这是不推荐直接做的,必须确保对象在运行时确实是子类的实例。否则,编译时即使允许,也会导致运行时抛出 ClassCastException

2.1 编译时父类,运行时子类:可以强制转换

如果声明时是父类引用,但运行时对象类型是子类,那么你可以通过强制转换将父类引用转换成子类类型,调用子类的特有方法。

Father f2 = new Son();  // 编译时是父类引用,运行时是子类实例
f2.run();  // 调用父类方法,OK((Son)f2).study();  // 强制转换,调用子类方法

解释:

  • 这里,f2 在编译时是 Father 类型,但它实际上是指向 Son 类型的对象(多态)。因此,你可以强制转换 f2Son 类型,调用 Son 特有的方法 study()
  • 需要注意的是,在进行强制转换之前,你应该确保 f2 实际上指向的是 Son 类型的对象。否则会抛出 ClassCastException

2.2 编译时父类,运行时也是父类:不可以强制转换

如果编译时是父类引用,且运行时对象也是父类类型,那么你不能强制转换为子类类型,转换会失败。

Father f3 = new Father();  // 编译时和运行时都是父类
((Son)f3).study();  // 强制转换会抛出 ClassCastException

解释:

  • 这里,f3 实际上是 Father 类型的对象,不能强制转换为 Son 类型。这种情况下,会抛出 ClassCastException,因为 Father 类和 Son 类没有直接的关系(Father 并不是 Son 的父类)。

总结:

  1. 小转大:子类转父类(向上转型)
    • 允许:子类对象可以赋值给父类引用。
    • 调用父类的方法,不会调用子类特有的方法。
  2. 大转小:父类转子类(向下转型)
    • 如果编译时是父类引用,运行时是子类对象,则可以通过强制转换调用子类的方法(多态情况下可以转换)。
    • 如果编译时是父类引用,运行时也是父类对象,则无法进行转换,强转会抛出 ClassCastException

ClassCastException 是 运行时异常(Runtime Exception)。

异常类别

在 Java 中,异常大致分为两类:

  1. 检查型异常(Checked Exception)
    • 这些异常是在编译时被检查的,程序必须显式处理这些异常(通过 try-catchthrows 声明)。
    • 例如:IOExceptionSQLException 等。
  2. 非检查型异常(Unchecked Exception)
    • 这些异常是在运行时发生的,编译器不会强制要求处理这些异常(但可以选择处理)。
    • RuntimeException 是所有非检查型异常的父类。
    • 例如:NullPointerExceptionArrayIndexOutOfBoundsExceptionClassCastException 等。

ClassCastException 的具体含义

ClassCastException 是 Java 中的 运行时异常(Unchecked Exception),通常在你尝试将一个对象强制转换为不兼容的类型时抛出。它表示你在运行时尝试进行不合法的类型转换。

举个例子:

Object obj = new String("Hello");
Integer num = (Integer) obj;  // 这里会抛出 ClassCastException

在这个例子中,obj 被声明为 Object 类型,但它实际上指向的是一个 String 对象。你尝试将其强制转换为 Integer 类型时,Java 会抛出 ClassCastException,因为 StringInteger 是不兼容的类型。

为什么是运行时异常?

ClassCastException运行时异常,意味着编译器不会检查类型转换是否安全,直到程序运行时才会发现转换的类型不兼容。如果你在编译时尝试转换为不兼容的类型,编译器不会报错,而是等到程序运行时抛出这个异常。

总结:

  • ClassCastException 是运行时异常,继承自 RuntimeException
  • 它表示你在运行时尝试进行不合法的类型转换,通常是强制转换时出现的错误。

instanceof

无法直接判断两个类型之间是否可以相互转换,只能用来判断一个对象能否安全地转换为某个类型

class Person {public void run() {System.out.println("Person is running");}
}class Student extends Person {public void study() {System.out.println("Student is studying");}
}class Teacher extends Person {public void teach() {System.out.println("Teacher is teaching");}
}public class Test {public static void main(String[] args) {Person p1 = new Student();  // p1 是一个 Student 类型的对象Person p2 = new Teacher();  // p2 是一个 Teacher 类型的对象Person p3 = new Person();   // p3 是一个 Person 类型的对象// 判断 p1 是否可以转换为 Studentif (p1 instanceof Student) {Student s = (Student) p1;  // 安全转换,因为 p1 实际上是 Student 类型s.study();  // 调用子类特有方法}// 判断 p2 是否可以转换为 Studentif (p2 instanceof Student) {Student s = (Student) p2;  // 这个不会执行,因为 p2 实际上是 Teacher 类型s.study();}// 判断 p3 是否可以转换为 Studentif (p3 instanceof Student) {Student s = (Student) p3;  // 这个不会执行,因为 p3 实际上是 Person 类型s.study();}}
}

转化部分 CAL总结:

1.小转大 子类转换成父类 随便转 run 是父类Father 中独有的方法

Son s1 = new Son();
Father f1 =s1;  
f1.run();   //或者直接 如第四行
((Father)s1).run();

2.大转小就比较麻烦了 ,主要还是看对象运行时的类型

​ 2.1 如果编译时父类 运行时是子类,那么可以强转 (多态) 但是还是可能出现问题 ClassCastException

​ 2.2 如果编译时父类 运行时还是父类 这就不可以强转

多态的关键在于:编译时的类型决定了引用变量的可用方法(即只能访问父类或实现接口中声明的方法),而运行时的类型决定了实际执行哪个方法。

父类 Person

package com.oop.demo07;public class Person {public Person() {}public void run(){System.out.println("Running Person");}//public void study(){//}//重写之后启动类中的第11行可以使用了,不报错 也就是说父类对象可以是用子类重写的方法}

子类 Student

package com.oop.demo07;public class Student extends Person{public Student() {}public void study(){System.out.println("Study");}
}

启动类Application

package com.oop.demo07;public class Application {public static void main(String[] args) {Student s1 = new Student();s1.run();s1.study();//((Person)(s1)).study();//已经转成父类,可能就有部分子类方法丢失了((Person)(s1)).run();Person p1 =s1; //子类转换成父类 直接转p1.run();  //第10行和第11行效果同第9行  只是新建了一个父类Persond实例p1System.out.println("========================");Person s2 = new Student();s2.run();//s2.study();//父类不能用子类独有的方法,需要重写在父类中声明一样的方法//如果不重写方法就通过强制转换,将父类对象转换成子类对象,调用子类中的方法((Student)s2).study();//将编译时父类Person(声明时)的S2 转换成子类Stendent类((Student)s2).run();//转换成子类 同样可以继承父类的run()方法System.out.println("========================");Person s3 = new Person();s3.run();//子类可直接转换成父类 小转大 但是大转小 父类转子类//需要多态声明 主要看运行是对象的类型 像 26行27行// 这样直接定义父类 (声明时(编译)是父类,运行时(实例类型)也是父类)将父类强转成子类是不可以的//((Student)s3).study(); //报错//((Student)s3).run();//报错}
}

输出结果:

Running Person
Study
Running Person
Running Person
========================
Running Person
Study
Running Person
========================
Running Person

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/868326.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【Python基础】Python虚拟环境

什么是Python环境 要搞清楚什么是虚拟环境,首先要清楚Python的环境指的是什么。当我们在执行python test.py时,思考如下问题:python哪里来? 这个主要归功于配置的系统环境变量PATH,当我们在命令行中运行程序时,系统会根据PATH配置的路径列表依次查寻是否有可执行文件pyth…

地平线 LiDAR-Camera 融合多任务 BEVFusion 参考算法-V1.0

该示例为参考算法,仅作为在 征程 6 上模型部署的设计参考,非量产算法。1.简介 激光雷达天然地具有深度信息,摄像头可以提供丰富的语义信息,它们是车载视觉感知系统中两个最关键的传感器。但是,如果激光雷达或者摄像头发生故障,则整个感知框架不能做出任何预测,这在根本上…

47. web框架

1. web框架本质 web框架本质上可以看成一个功能强大的socket服务端,用户的浏览器可以看成拥有可视化界面的socket客户端 通过网络请求实现数据交互,可简单的将web框架看做是对前端、数据库的全方位整合2. 手写简易版web框架 服务端import socketserver = socket.socket() ser…

记一次生产事故:一年的数据被删除了

故事开始快要下班了,心情是这样的。 突然电话响了,一看电话号码是项目甲方负责人的号码菊花一紧,难道出什么问题了,硬着头皮接通了电话。@####@,你们程序怎么回事 ,某个业务的数据界面一条都查询不到了。。。。。我们马上排查一下,看看那什么情况(心理活动:尼玛,尼玛…

【Azure App Service】对App Service中CPU指标数据中系统占用部分(System CPU)的解释

问题描述 在使用Azure App Service服务时,对于它的CPU占比,从App Service Plan级别可以查看整个实例(vm)资源的CPU占比,而如果在具体的一个App Service服务中,则只能查看到当前应用所消耗的CPU时间,如果需要计算它的占比:需要使用公式【 CPU Time / (CPU核数 * 60 )】估算…

2024秋季学期 光学期末复习笔记

累了,懒得写文案了参考资料 [1] 赵凯华, 钟锡华. 光学(重排本)[M]. 北京大学出版社, 2017.10. [2] 崔宏滨, 李永平, 康学亮. 光学(第二版)[M]. 科学出版社, 2015.7. [3] 王安廷. 光学课程PPT[Z]. 中国科学技术大学, 2024. 干涉衍射偏振

Python 基础知识 之 选择(分支)结构 + 模式匹配结构(match)

选择结构按照条件选择执行不同的代码段1. 单分支结构if语法结构执行流程:如果表达式的值为True,就执行语句块,如果表达式的值为False,就跳过语句块,继续执行下面的语句 ⭐注意: ⭐⭐⭐表达式后面的冒号; 缩进,python中通过缩进来控制程序逻辑示例;# 1. 判断是否中奖 n…

中考英语优秀范文-热点话题-传统文化-003 Chinese Tea 中国茶

1 写作要求 第三次法国国际友好交流学校线上论坛拟于下周五举行。本期论坛的主题为“茶文化”,作为论坛特邀嘉宾,请你写一篇文章,向国际友好交流学校介绍中国的特色茶饮和中国茶文化。 内容包括: 1 茶的历史(中国人种茶、饮茶已有4000多年的历史)和种类(绿茶、红茶、乌龙…

Hetao P3804 Cut 题解 [ 蓝 ] [ AC 自动机 ] [ 差分 ]

AC 自动机简单题。Cut:AC 自动机简单题。思路 看见多个模式串以及求前缀,很显然能想到构建一个 AC 自动机。 那么在用 \(T\) 查询时,当前指针的深度就是该位置的最长前缀匹配长度。这个在字典树 insert 的时候就能求出来。 求出每一位的最长前缀后,因为这些部分都不能作为分…

Lec 14 文件系统与设备

Lec 14 文件系统与设备License 本内容版权归上海交通大学并行与分布式系统研究所所有 使用者可以将全部或部分本内容免费用于非商业用途 使用者在使用全部或部分本内容时请注明来源 资料来自上海交通大学并行与分布式系统研究所+材料名字 对于不遵守此声明或者其他违法使用本内…

Java基础学习(七)

Java基础学习(七):异常机制 目录Java基础学习(七):异常机制概念简单分类异常体系结构异常处理机制自定义异常 本文为个人学习记录,内容学习自 狂神说Java概念异常指程序运行中出现的不期而至的各种状况,如:文件找不到、网络连接失败、非法参数等 异常发生在程序运行期…

24. 基于项的控件

一、基于项的控件PySide6 有专门的显示数据的控件和存储数据的模型,可以显示和存储不同形式的数据。显示数据的控件分为两类,一类是基于 项(item)的控件,另一类是基于 模型(model)的控件,基于项的控件是基于模型的控件的简便类。基于项的控件把读取到的数据存储到项中,…