RError.com

RError.com Logo RError.com Logo

RError.com Navigation

  • 主页

Mobile menu

Close
  • 主页
  • 系统&网络
    • 热门问题
    • 最新问题
    • 标签
  • Ubuntu
    • 热门问题
    • 最新问题
    • 标签
  • 帮助
主页 / 问题 / 629700
Accepted
Katrin Mayer
Katrin Mayer
Asked:2020-02-19 05:46:27 +0000 UTC2020-02-19 05:46:27 +0000 UTC 2020-02-19 05:46:27 +0000 UTC

父类和子类的类(构造函数)的初始化顺序,两个类中都有多个构造函数

  • 772

对于类继承的情况,我发现了以下对象初始化顺序:

  • 父类的静态字段;
  • 父类的静态初始化块;
  • Child 类的静态字段;
  • Child类的静态初始化块;
  • 父类的非静态字段;
  • 父类的非静态初始化块;
  • 父类构造函数;
  • Child 类的非静态字段;
  • 类 Child 的非静态初始化块
  • 子类构造函数。

但是,如果父项和子项中都有多个构造函数怎么办——它们是如何加载的?parent 的所有构造函数,然后是 successor,还是按书写顺序一次一个?

在面试中,有人向我展示了一个示例,说明继承人的构造函数(或其执行 - 我记不清了)可能由于父母构造函数的顺序而出现错误,但我无法在任何地方找到明确的信息并且代码中这种情况的示例。你能澄清这一点吗?

java
  • 2 2 个回答
  • 10 Views

2 个回答

  • Voted
  1. Best Answer
    Regent
    2020-02-19T16:09:06Z2020-02-19T16:09:06Z

    在派生类中,调用父类的一个构造函数。
    要么在子构造函数的开头使用 construct 显式调用父构造函数,要么在编译器在构造函数的开头super替换它时隐式调用它。super();而且,如果父类没有不带参数的构造函数,那么代码会导致编译错误。
    有关此的文档。

    因此,对于有多少父构造函数以及哪些在子构造函数中被调用是没有不确定性的。

    正如@zRrr 所写,如果在父构造函数中调用重写或子实现的方法,而父构造函数又使用尚未初始化的子字段,那么调用构造函数的顺序和类字段的初始化可能会出现问题。
    例子:

    public abstract class Animal
    {
        private final String fullName;
    
        protected Animal()
        {
            fullName = "Animal " + getName();
        }
    
        public String getFullName()
        {
            return fullName;
        }
    
        protected abstract String getName();
    }
    
    public class Dog extends Animal
    {
        private String thisName = "DDog";
    
        protected String getName()
        {
            return thisName;
        }
    }
    
    public static void main(String[] args)
    {
        Animal animal = new Dog();
        System.out.println(animal.getFullName());
    }
    

    它会显示在屏幕上Animal null,因为在调用该方法的那一刻,getNameyDog字段thisName还没有被初始化。
    创建animal时,默认构造函数 y 被调用Dog,这首先导致调用不带参数的构造函数 y Animal,然后才初始化该字段thisName。


    这篇文章还有另一个例子:

    public class Upper
    {
        String upperString;
    
        public Upper()
        {
            Initializer.initialize(this);
        }
    }
    
    public class Lower extends Upper
    {
        String lowerString = null;
    
        public Lower()
        {
            super();
            System.out.println("Upper:  " + upperString);
            System.out.println("Lower:  " + lowerString);
        }
    
        public static void main(final String[] args)
        {
            new Lower();
        }
    }
    
    public class Initializer
    {
        static void initialize(final Upper anUpper)
        {
            if (anUpper instanceof Lower)
            {
                Lower lower = (Lower)anUpper;
                lower.lowerString = "lowerInited";
            }
            anUpper.upperString = "upperInited";
        }
    }
    

    屏幕将显示:

    Upper:  upperInited
    Lower:  null;
    

    但是,如果您替换String lowerString = null;为String lowerString;,则输出将是:

    Upper:  upperInited
    Lower:  lowerInited
    

    原始版本的问题是对的赋值null发生String lowerString = null;在调用父类的构造函数之后,而父类的构造函数又调用Initializer.initialize. 也就是说,lowerString首先写的是"lowerInited",然后是null。


    这两个示例都说明了由于对代码执行顺序的错误假设而引起的问题。很难理解这段代码中执行的是什么,这本身就表明在实践中不应该这样写代码。

    • 7
  2. Aliaksandr Shpak
    2020-07-24T21:12:51Z2020-07-24T21:12:51Z

    我认为这是顺序:

    /**
     * Порядок инициализации таков:
     * [1]. Статические поля базового класса;
     * [2]. Статический блок инициализации базового класса;
     * [3]. Статические поля производного класса;
     * [4]. Статический блок инициализации производного класса;
     * [5]. Поля (Глобальные переменные) базового класса;
     * [6]. Нестатический блок инициализации базового класса;
     * [7]. Конструктор базового класса [если у родителя нет конструктора без аргументов, то код приведёт к ошибке компиляции];
     * [8]. Поля (Глобальные переменные)  производного класса;
     * [9]. Нестатический блок инициализации производного класса;
     * [10]. Конструктор производного класса;
     */
    
    class Base {
    
        //[1] transient переменные и статические поля не сериализуются
        private static Base instance = new Base("))(("); //BASE = 0 //
        public static int _baseStaticInsect = 111; //сокрыто, доступ Base._baseStaticBase
        private static final int DELTA = 2;
        private static int BASE = 3;
    
        //[2]
        static {
            System.out.println("[static instance class initializer] " + Inherit.class.getSimpleName());
        }
    
        //[5]
        private int i = 5;
        protected int j; //mod:protected-internal
        int x;
        int internal; //mod:package
        public int iInc;
    
        //[6]
        {
            System.out.println("[instance class initializer] " + this.getClass().getSimpleName());
        }
    
    
        //[7]
        //private недоступен  из вне, нельзя создать без рефлексии
        //если у нет конструктора без аргументов или он приватный, то код приведёт к ошибке компиляции];
        public Base() {
            System.out.println("[constructor] " + this.getClass().getSimpleName());
            x = BASE + DELTA;
        }
    
        //[7]
        public Base(String data) {
            this();
            System.out.println("[constructor] (String)" + this.getClass().getSimpleName());
            System.out.println(data);
    
        }
    
        static int printInit(String s) {
            return -1;
        }
    }
    
    public class Inherit extends Base {
    
        //[3]
        public static int _baseStaticBase = 222;
        public static int _baseStaticInherit = 4;
    
        //[4]
        static {
            System.out.println("[static class initializer] " + Inherit.class.getSimpleName());
        }
    
        //[8]
        private int internal;
    
        public int Integnal() { //getInternal
            return this.internal;
        }
    
        private boolean internalFlag;
    
        public boolean isInternalFlag() {
            return internalFlag;
        }
    
        //[9]
        {
            System.out.println("[instance class initializer] Inherit " + this.getClass().getSimpleName());
        }
    
        //[10]
        public Inherit() {
            this("start");
            System.out.println("Inherit constructor");
        }
    
        //[10]
        public Inherit(String data) {
            System.out.println("Inherit constructor data");
        }
    
        public static void main(String[] args) {
            Base base = new Base();
            Inherit inherit = new Inherit();
        }
    
    }
    
    • 1

相关问题

Sidebar

Stats

  • 问题 10021
  • Answers 30001
  • 最佳答案 8000
  • 用户 6900
  • 常问
  • 回答
  • Marko Smith

    Python 3.6 - 安装 MySQL (Windows)

    • 1 个回答
  • Marko Smith

    C++ 编写程序“计算单个岛屿”。填充一个二维数组 12x12 0 和 1

    • 2 个回答
  • Marko Smith

    返回指针的函数

    • 1 个回答
  • Marko Smith

    我使用 django 管理面板添加图像,但它没有显示

    • 1 个回答
  • Marko Smith

    这些条目是什么意思,它们的完整等效项是什么样的

    • 2 个回答
  • Marko Smith

    浏览器仍然缓存文件数据

    • 1 个回答
  • Marko Smith

    在 Excel VBA 中激活工作表的问题

    • 3 个回答
  • Marko Smith

    为什么内置类型中包含复数而小数不包含?

    • 2 个回答
  • Marko Smith

    获得唯一途径

    • 3 个回答
  • Marko Smith

    告诉我一个像幻灯片一样创建滚动的库

    • 1 个回答
  • Martin Hope
    Air 究竟是什么标识了网站访问者? 2020-11-03 15:49:20 +0000 UTC
  • Martin Hope
    Алексей Шиманский 如何以及通过什么方式来查找 Javascript 代码中的错误? 2020-08-03 00:21:37 +0000 UTC
  • Martin Hope
    Qwertiy 号码显示 9223372036854775807 2020-07-11 18:16:49 +0000 UTC
  • Martin Hope
    user216109 如何为黑客设下陷阱,或充分击退攻击? 2020-05-10 02:22:52 +0000 UTC
  • Martin Hope
    Qwertiy 并变成3个无穷大 2020-11-06 07:15:57 +0000 UTC
  • Martin Hope
    koks_rs 什么是样板代码? 2020-10-27 15:43:19 +0000 UTC
  • Martin Hope
    user207618 Codegolf——组合选择算法的实现 2020-10-23 18:46:29 +0000 UTC
  • Martin Hope
    Sirop4ik 向 git 提交发布的正确方法是什么? 2020-10-05 00:02:00 +0000 UTC
  • Martin Hope
    faoxis 为什么在这么多示例中函数都称为 foo? 2020-08-15 04:42:49 +0000 UTC
  • Martin Hope
    Pavel Mayorov 如何从事件或回调函数中返回值?或者至少等他们完成。 2020-08-11 16:49:28 +0000 UTC

热门标签

javascript python java php c# c++ html android jquery mysql

Explore

  • 主页
  • 问题
    • 热门问题
    • 最新问题
  • 标签
  • 帮助

Footer

RError.com

关于我们

  • 关于我们
  • 联系我们

Legal Stuff

  • Privacy Policy

帮助

© 2023 RError.com All Rights Reserve   沪ICP备12040472号-5