Java 不要在父类的构造方法里面调用可以被子类重写的方法

不要在父类的构造方法(代码块)里面调用可以被子类重写的方法

我们从第一天学习Java开始,就对Java的类初始化顺序牢记于心。但是在实际开发过程中,似乎很难能接触这一部分的应用。在这之前,我也认为它只是面试中八股文而已,直到最近踩了一个坑,这才发现它是多么的重要。

1. 说说坑?

地球人都知道,在Java世界中,类初始化顺序是:

  • 父类的 静态成员变量 或者 静态代码块 --> 子类的 静态成员变量 或者 静态代码块 -->
  • 父类 成员变量 或者 普通代码块 --> 父类的 构造方法 --> 子类 成员变量 或者 普通代码块 --> 子类的 构造方法。

我们在理解上都没有问题,但是使用上很容易忽略这个顺序。我举一个例子:假设有两个类,分别是AnimalDog类,其中Dog继承于Animal(是不是感觉回到了学习Java的第一天?)。它们的代码如下:

public class Animal {public Animal() {eat();}protected void eat() {}
}// ----------------------------
public class Dog extends Animal {private String foodName = "肉";public Dog() {}@Overrideprotected void eat() {System.out.println("我吃" + foodName);}
}

代码实现很简单,总的来说就是:父类有一个非final的方法在其构造方法里面回调,子类重写了这个方法,同时在这方法里面访问了一个自己的一个成员变量。

那么这里会什么问题呢?大家可以运行一下上面的代码,发现输出的结果是:我吃null,也就是说foodName还没有初始化。在这里,很容易理解为什么foodName没有初始化,因为在类初始化顺序中,子类的成员变量在父类的构造方法之后才初始化。这个Demo足够的简单,所以大家觉得没有问题,在正式的开发环境中,情况一般都是非常复杂的,代码成千上万,我们在一个方法里面访问本类的成员变量,一般不会关心这个方法是什么时候回调的,从而导致在开发过程中遇到不可预知的结果。

不过幸运的是,这种问题是必现的,一般在开发中就能遇到并且解决,并不会带到线上去,但是我还是想说:父类的构造方法(代码块)里面不要调用可以被子类重写的方法

除此以外,这种调用方式还会遇到一个隐式的问题:

public class Dog extends Animal {private String foodName1;private String foodName2 = null;public Dog() {}@Overrideprotected void eat() {foodName1 = "肉";foodName2 = "骨头";}public void print() {System.out.println("foodName1 = " + foodName1 + " foodName2 = " + foodName2);}
}

我改写了Dog类的代码,在其内部定义了两个成员变量,分别是:foodName1foodName2,它们都在eat方法里面赋值,需要注意的是,foodName2在定义设置默认值为null,而foodName1没有设置默认值。那么我们创建Dog的对象,然后调用它的print方法,结果如何呢?结果是:foodName1 = 肉 foodName2 = null。也就是说,设置默认值的成员变量即使在方法里面重新赋值,在真正使用的时候却还是默认值。这是为啥呢?其实要想真正了解其内部原理,就要扒Dog类的字节码了,但是我懒得扒,猜测其原因是:如果在定义成员变量没有设置默认值时,当其初始化时发现已经被初始化,就不再进行初始化。 换言之,如果设置了默认值,即使之前已经被赋值,也会被默认值覆盖。

相比于第一个坑,第二个坑显得更加隐式和不容易理解。总之还是那么一句话:父类的构造方法(代码块)里面不要调用可以被子类重写的方法。最重要的是,在这个过程中,编译器并没有做出任何提示,这也是这类问题容易出现的原因之一吧。

2. 聊聊Kotlin?

也许大家在Java中习惯了上面的使用方式,同时编译器也没有报错,说明官方也默认这种操作(我乱说的)。但是,我们以相同的方式应用在Kotlin中,编译器报了一个警告:

open class Test {private val string = getString()protected open fun getString(): String {return ""}
}

在这里插入图片描述

总体上来说,比Java友好的多,至少有了一个警告提示。

从这里,我们可以得出来两个结论:

  1. 不要随意忽略编译器给出的警告。
  2. 快去使用Kotlin吧!!!

补充:关于Kotlin中这个问题的视频讲解:[Kotlin] 为什么父类构造器不建议调用 open 的函数?(其实跟Java本质上是同样的问题)

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

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

相关文章

聊一聊大模型 | 京东云技术团队

事情还得从ChatGPT说起。 2022年12月OpenAI发布了自然语言生成模型ChatGPT,一个可以基于用户输入文本自动生成回答的人工智能体。它有着赶超人类的自然对话程度以及逆天的学识。一时间引爆了整个人工智能界,各大巨头也纷纷跟进发布了自家的大模型&#…

Python 潮流周刊#29:Rust 会比 Python 慢?!

△请给“Python猫”加星标 ,以免错过文章推送 你好,我是猫哥。这里每周分享优质的 Python、AI 及通用技术内容,大部分为英文。本周刊开源,欢迎投稿[1]。另有电报频道[2]作为副刊,补充发布更加丰富的资讯。 &#x1f43…

算法基础--双指针

前面已经写了两篇关于算法方面的文章,这几天想了下,决定把这个算法整理成一个系列,除了是帮助自己巩固算法知识外,还能够把自己总结的每种算法的套路保存下来并分享给大家,这样以后即使是哪天想要重拾起来,…

简单可行的SeruatV4的安装方案

目前Seurat的版本从V4升级到了V5,由于一些变化,导致当年取巧,使用获取数据的方法都无法在V5中使用。 建议在操作前重启下Rstudio(或更确切的说是R)!!! 那么如何确保自己能够安装V4的…

【字符串匹配】【KMP算法】Leetcode 28 找出字符串中第一个匹配项的下标☆

【字符串匹配】【KMP算法】Leetcode 28 找出字符串中第一个匹配项的下标 (1)前缀和后缀(2)前缀表(最长相同的前缀和后缀的长度)(3)匹配过程示意(4)next数组的…

matlab 点云放缩变换

目录 一、算法原理二、代码实现三、结果展示四、相关链接本文由CSDN点云侠原创,原文链接。爬虫网站自重。如果你不是在点云侠的博客中看到该文章,那么此处便是不要脸的爬虫与GPT。 一、算法原理 缩放可以独立应用于三个坐标轴,如将点 ( x , y , z ) ( x

[LeetCode周赛复盘] 第 374 场周赛20231203

[LeetCode周赛复盘] 第 374 场周赛20231203 一、本周周赛总结100144. 找出峰值1. 题目描述2. 思路分析3. 代码实现 100153. 需要添加的硬币的最小数量1. 题目描述2. 思路分析3. 代码实现 100145. 统计完全子字符串1. 题目描述2. 思路分析3. 代码实现 100146. 统计感冒序列的数…

使用Linux docker方式快速安装Plik并结合内网穿透实现公网访问

文章目录 1. Docker部署Plik2. 本地访问Plik3. Linux安装Cpolar4. 配置Plik公网地址5. 远程访问Plik6. 固定Plik公网地址7. 固定地址访问Plik 本文介绍如何使用Linux docker方式快速安装Plik并且结合Cpolar内网穿透工具实现远程访问,实现随时随地在任意设备上传或者…

C语言扫雷游戏

提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言一、扫雷游戏的分析和设计1.1扫雷游戏的功能说明1.2数据结构的分析1.3文件结构设计 二、扫雷游戏的代码实现总结 前言 详细介绍扫雷游戏的思路和实现过程。 一…

高校人员信息管理系统C++

代码:https://mbd.pub/o/bread/ZZeZk5lx 一、基本内容论述 1、问题描述 某高校有四类员工:教师、实验员、行政人员、教师兼行政人员;共有的信息包括:编号、姓名、性别、年龄等。其中,教师还包含的信息有:所…

堆排序(C语言)

前言 在上一篇内容:大小堆的实现(C语言),我们实现了关于创建大小堆的各函数与实现。但是如果突然要使用一个堆排序但是此时并没有一个现成的堆,这就需要花费时间去新建实现堆的插入删除这些操作从而实现一个堆&#xf…

51单片机应用从零开始(十)·指针

指针 C语言指针是一种保存变量地址的数据类型。它可以让程序直接访问内存中的数据,而不需要通过变量名来访问。指针变量存储的是一个地址,这个地址指向内存中的某个位置,该位置存储了一个值。 在C语言中,可以使用&运算符取得一…

Endnote加入新的style(参考文献格式)

1. 下载模板 可以从官网上下载模板,比如某些常见的期刊都有自己的模板,还有写中文论文的话有专门的GBT7714。 2. 示范 以下图MDPI为例,下载下来是一个ens文件。 双击打开此文件 file -> save as 输入保存的名字,我这里保…

字符串转换整数

字符串转换整数 描述 : 请你来实现一个 myAtoi(string s) 函数,使其能将字符串转换成一个 32 位有符号整数(类似 C/C 中的 atoi 函数)。 函数 myAtoi(string s) 的算法如下: 读入字符串并丢弃无用的前导空格检查下一个字符&am…

Linux:docker镜像的创建(5)

1.基于已有镜像创建 步骤: 1.将原始镜像加入容器并运行 2.在原始镜像中部署各种服务 3.退出容器 4.使用下面命令将容器生成新的镜像 现在我们在这个容器里做了一些配置,我们要把他做成自己镜像 docker commit -m "centos7_123" -a "tarr…

【工具使用-Audition】如何使用Audition查看频率

一,简介 在工作过程中要对处理后的音频进行频率分析,本文以Audition 2020为例进行说明,供参考。 二,操作步骤 2.1 生成测试音源 使用Audition生成左通道为1KHz,右通道为10KHz的音源信号 如图所示: 2.…

Android Init系统:引领设备启动的先锋

Android Init系统:引领设备启动的先锋 引言 Init系统是一个操作系统启动的必要组件,负责在启动时初始化所有系统资源、服务和应用程序。在Android设备中,Init系统起到了至关重要的作用,它是启动过程中的第一个进程,负…

学习知识回顾随笔(远程连接MySQL|远程访问Django|HTTP协议|Web框架)

文章目录 如何远程连接MySQL数据库1.创建用户来运行,此用户从任何主机连接到mysql数据库2.使用IP地址来访问MySQL数据库 如何远程访问Django项目Web应用什么是Web应用应用程序的两种模式Web应用程序的优缺点 HTTP协议(超文本传输协议)简介HTT…

C++-模板

目录 一.泛型编程 二.模板的分类 三.函数模板 1.函数模板的概念 2.函数模板格式 3.函数模板的原理 4.函数模板的实例化 a.隐式实例化 b.显式实例化 5.模板参数的匹配原则 四.类模板 1.类模板的定义格式 2.类模板的实例化 五.class和typename的区别 六.非类型模板…

路由策略,gRPC 路由如何实现

目录 一、为啥我们要路由策略: 二、基于gRPC 路由策略 一、为啥我们要路由策略: 我们可以重新回到调用方发起 RPC 调用的流程。在 RPC 发起真实请求的时候,有一个步骤就是从服务提供方节点集合里面选择一个合适的节点(就是我们…