子类重写父类变量_为什么在子类中不重写超类的实例变量

子类重写父类变量

当我们在父类和子类中创建一个具有相同名称的变量,并尝试使用持有子类对象的父类引用访问它时,我们会得到什么?

为了理解这一点,让我们考虑下面的示例,在该示例中,我们在ParentChild类中都声明了一个具有相同名称的变量x

class Parent {// Declaring instance variable by name `x`String x = "Parent`s Instance Variable";public void print() {System.out.println(x);}
}class Child extends Parent {// Hiding Parent class's variable `x` by defining a variable in child class with same name.String x = "Child`s Instance Variable";@Overridepublic void print() {System.out.print(x);// If we still want to access variable from super class, we do that by using `super.x`System.out.print(", " + super.x + "\n");}
}

现在,如果我们尝试使用以下代码访问x ,将打印什么System.out.println(parent.x)

Parent parent = new Child();
System.out.println(parent.x) // Output -- Parent`s Instance Variable

通常,我们会说Child类将覆盖Parent类中声明的变量,并且parent.x将给我们任何Child's对象所持有的东西。 因为在方法上进行相同类型的操作时发生的是同一件事。

但实际上并非如此, parent.x将为我们提供在Parent类中声明的Parent实例变量的值,但是为什么呢?

因为Java中的变量不遵循多态性,所以重写仅适用于方法,而不适用于变量。 并且,当子类中的实例变量与父类中的实例变量具有相同的名称时,则从引用类型中选择该实例变量。

在Java中,当我们在Child类中使用已经用于在Parent类中定义变量的名称定义变量时,Child类的变量将隐藏父类的变量,即使它们的类型不同。 这种概念称为可变隐藏。

换句话说,当子类和父类都具有相同名称的变量时,子类的变量将隐藏父类的变量。 您可以在文章什么是Java中的变量阴影和隐藏中阅读有关变量隐藏的更多信息。

变量隐藏与方法覆盖不同

尽管变量隐藏看起来像是覆盖变量,类似于方法覆盖,但事实并非如此,但覆盖仅适用于方法,而隐藏适用于变量。

方法覆盖的情况下,覆盖方法完全替换了继承的方法,因此当我们尝试通过持有子对象来从父对象的引用访问该方法时,将调用子类中的方法。 您可以在“方法重载与方法重载”一书中了解有关重载以及被重载的方法如何完全替代继承的方法的更多信息,以及为什么要遵循方法 重载 规则 。

但是在变量隐藏中,子类隐藏继承的变量而不是替换它们,这基本上意味着子类的对象包含两个变量,而子变量则隐藏了父变量。 因此,当我们尝试从Child类中访问变量时,将从子类中访问该变量。

如果我简化了示例8.3.1.1-3。 隐藏 Java语言规范 的实例变量 :

当我们在Child类中声明一个具有相同名称的变量时,例如x作为Parent类中的实例变量,则

  1. 子类的对象包含两个变量(一个是从Parent类继承的,另一个是在Child本身中声明的),但是子类变量隐藏了父类的变量。
  2. 由于声明xChild皮的定义xParent ,类的声明中Child ,简单名称x总是指到外地类中声明的Child 。 并且,如果Child类方法中的代码想要引用Parent类的变量x ,则可以将其作为super.x来完成。
  3. 如果我们尝试访问ParentChild类之外的变量,则从引用类型中选择实例变量。 因此,以下代码中的表达式parent2.x给出了属于父类的变量值,即使它持有Child的对象,但((Child) parent2).x可以从Child类访问该值,因为我们进行了相同的转换参考Child

实例变量

为什么以这种方式设计可变隐藏

因此我们知道实例变量是从引用类型而不是实例类型中选择的,并且多态性不适用于变量,但是真正的问题是为什么? 为什么变量被设计为跟随隐藏而不是覆盖。

因为如果我们在子类中更改其类型,则变量覆盖可能会破坏从父级继承的方法。

我们知道每个子类都从其父类继承变量和方法(状态和行为)。 想象一下,如果Java允许变量覆盖,并且我们在子类中将变量的类型从int更改为Object 。 它将破坏使用该变量的任何方法,并且由于子代已从父代继承了这些方法,因此编译器将在child类中给出错误。

例如:

class Parent {int x;public int increment() {return ++x;}public int getX() {return x;}
}class Child extends Parent {Object x;// Child is inherting increment(), getX() from Parent and both methods returns an int // But in child class type of x is Object, so increment(), getX() will fail to compile. 
}

如果Child.x覆盖Parent.xincrement()getX()工作? 在子类中,这些方法将尝试返回错误类型的字段的值!

如前所述,如果Java允许变量覆盖,则Child的变量不能替代Parent的变量,这将破坏Liskov替代性原则(LSP)。

为什么从引用类型而不是实例中选择实例变量

如JVM内部如何处理方法重载和覆盖中所述 ,在编译时,覆盖方法调用仅从引用类处理,但是所有覆盖的方法在运行时都使用vtable被覆盖方法替代,这种现象称为运行时多态性。

同样,在编译时也从引用类型处理变量访问,但是正如我们所讨论的那样,变量不遵循重写或运行时多态性,因此它们在运行时不会被子类变量替代,仍然引用引用类型。

一般来说,没有人会建议隐藏字段,因为这会使代码难以阅读并造成混乱。 如果我们始终坚持下去,这种混乱就不会出现。
创建POJO并通过声明为私有字段来封装我们的字段的一般准则,并根据需要提供getter / setter,以便在该类之外看不到变量,并且子类无法访问它们。

您可以在此Github存储库中找到完整的代码,请随时提供宝贵的反馈。

翻译自: https://www.javacodegeeks.com/2018/11/instance-variable-class-overridden-class.html

子类重写父类变量

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

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

相关文章

Spring基于 XML 的声明式事务控制(配置方式)

一、引入依赖 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http://maven.apache.org/POM/4…

javadoc - Java API 文档生成器(Windows版本)

文章目录简介命令语法结构Javadoc Doclets术语带文档的类引用类外部引用类源文件源代码文件包注释文件概述注释文件其他未处理文件生成的文件基本内容页交叉参考页支持文件HTML 框架生成的文件结构文档注释注释源代码JAVADOC 标记author name-textdeprecated deprecated-textex…

光缆故障测试_简单的测试可以防止最严重的故障

光缆故障测试错误处理是软件开发中最困难且被忽略的部分之一&#xff0c;而且如果系统是分布式的&#xff0c;那么这将变得更加困难。 不错的论文写在“ 简单测试可以预防最关键的故障” 主题上。 每个开发人员都应该阅读本文。 我将尝试总结本文的主要内容&#xff0c;但建议…

使用aop解决事务问题(xml版)

一、引入依赖 pom.xml代码&#xff1a; <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"xmlns:xsi"http://www.w3.org/2001/XMLSchema-instance"xsi:schemaLocation"http:…

印象大使_基本服务-使用大使网关

印象大使这是我对Knative服务进行实验的延续&#xff0c;这次是围绕在Knative服务应用程序之上构建网关。 这建立在我之前的两篇文章的基础上- 使用Knative部署Spring Boot App以及在Knative中进行服务到服务的调用 。 为什么要在Knative应用程序之上使用网关 为了解释这一点&…

Eclipse 如何生成项目的javadoc(API 文档)

文章目录生成 javadoc 的三种方法乱码问题生成 javadoc 的三种方法 1.在项目列表中按右键&#xff0c;选择 Export&#xff0c;然后在 Export 对话框中选择 Java 下的 Javadoc&#xff0c;然后点击 Next 最后按 Finish 提交即可开始生成文档。 2.用菜单选择&#xff1a;File…

使用注解版AOP解决事务问题

一、注解版和xml版的区别 1、 通知的四种常用类型 &#xff08;1&#xff09;aop:before 作用&#xff1a; 用于配置前置通知。指定增强的方法在切入点方法之前执行 属性&#xff1a; method:用于指定通知类中的增强方法名称 ponitcut-ref&#xff1a;用于指定切入点的表达式…

Windows系统下彻底卸载MySQL数据库

1.正常卸载MySQL数据库 2.将 C:/ProgramData 目录下的子目录 MySQL 删除 注意&#xff1a;目录 ProgramData 是隐藏目录&#xff0c;你正常情况下看不到&#xff0c;你可以在文件地址栏输入 C:/ProgramData 进入该目录下。

ibm收购red hat_IBM将收购Red Hat:面向Java的初衷

ibm收购red hatOracle公司近九年前完成了收购Sun Microsystems的过程。 那是当时的大新闻&#xff0c;今天也宣布了类似的大新闻&#xff1a; IBM和Red Hat已经同意IBM收购Red Hat 。 IBM主页面宣布“ IBM收购Red Hat”。 然后&#xff0c;它说&#xff1a;“这将改变一切。” …

Windows系统的MySQL目录结构

安装目录 安装 MySQL 时&#xff0c;如果我选择安装在 D:\develop\MySQL&#xff0c;那么你安装好之后点击文件夹 MySQL 可以看到如下内容&#xff1a; 目录 bin&#xff1a;存放二进制可执行文件 目录 data&#xff1a;存放数据库日志文件等 目录 include&#xff1a;C语言…

java监控数据库性能_Java:GraalVM数据库流性能

java监控数据库性能GraalVM是JVM块的新成员。 它是一个开源虚拟机&#xff0c;能够同时运行多种编程语言&#xff0c;例如Java&#xff0c;Rust和JavaScript。 GraalVM还具有一个新的内部代码优化器管道&#xff0c;与某些条件下的其他JVM相比&#xff0c;它可以显着提高性能。…

@Param注解注意事项(小的细节)

一、关于dao和UserDao.xml的参数问题 dao接口里是否有Param进行映射&#xff0c;和UserDao.xml的参数的书写格式有关系 多参数类型的时候&#xff0c;使用param 如果dao接口里写了Param&#xff08;“uid”&#xff09;&#xff0c;UserDao.xml的参数必须要写#{uid} 或者#{p…

jep122_JEP 181不兼容,嵌套类/ 2

jep122JEP 181是基于嵌套的访问控制https://openjdk.java.net/jeps/181 。 它是在Java 11中引入的&#xff0c;它故意引入了与先前版本的不兼容性。 这是一个很好的例子&#xff0c;与Java的早期版本兼容并不是刻板的规则&#xff0c;而是保持语言的一致性和稳定发展。 在本文中…

文档中快速输入日期时间的方法

文章目录word文档excel表格日期输入后固定不变日期输入后会根据系统变化的输入法快速输入日期时间&#xff08;推荐&#xff09;word文档 输入 ctrl;&#xff0c;就可以直接输入当前日期 输入 ctrlshift;&#xff0c;就可以直接输入当前时间 如果是输入当前日期和时间&#xf…

=空值返回空值_@ParameterizedTest在@CvsSource中具有空值

空值返回空值在JUnit 4中编写参数化测试非常麻烦。 JUnit 5对框架进行了一些有用的改进&#xff0c;并且使用不同的参数运行相同的测试比以前的版本要简单得多。 但是&#xff0c;在这样的参数中传递空值存在一个小问题。 在本文中&#xff0c;我将向您展示如何在JUnit 5中的P…

MySQL的配置文件_选项文件_参数文件

文章目录WindowsLinux使用命令 locate 查找含有关键词 my.cnf 的文件路径查看是否使用了指定目录下的 my.cnf 文件查看 MySQL 默认读取 my.cnf 文件路径启动时没有读取任何配置文件主要的配置参数类似于 Oracle 的参数文件&#xff0c;MySQL 的选项文件&#xff08;即配置文件&…

java url参数转换:_提示:通过URL激活并发送参数

java url参数转换:世界上最安全的密码是不存在的密码。 使用完全随机的密钥从等式中删除用户。 公平地说&#xff0c;这有一些缺点&#xff0c;并且密码仍然存在于某处&#xff08;在您的电话/电子邮件中&#xff09;&#xff0c;但通常这样效果很好... 诀窍很简单&#xff0c…

简单的vue入门案例

一、 简单入门Hello World案例 二、插值表达式 三、点击事件 四、按键事件 1.如果按下不是 0 - 9 则阻止事件执行 2.打印按下什么按键 五、鼠标事件 1.打印绝对坐标 2.打印相对坐标 3.鼠标mousemove 4. 阻止冒泡(阻止事件的传播) 六、事件修饰符 1.阻止跳转页面v-on:submit.…

javafx 调用java_Java,JavaFX的流畅设计风格进度栏

javafx 调用java按照承诺&#xff0c;刚刚发布的Java JavaFX主题JMetro版本4.6为进度栏带来了新样式。 进度栏有两种可能的状态&#xff1a;确定和不确定&#xff0c;新的JMetro版本针对这两种具有不同的样式。 在本文中&#xff0c;我还将详细介绍我在JMetro中遵守的一些API设…