Java类的完整构造执行顺序

news/2024/7/3 13:36:20 标签: java, object, hashmap, float, cache, 编译器

这里只说一个完整的结果,至于为什么是这样的顺序,可以参考我以前的文章:深入剖析java类的构造方式

  1. 如果父类有静态成员赋值或者静态初始化块,执行静态成员赋值和静态初始化块
  2. 如果类有静态成员赋值或者静态初始化块,执行静态成员赋值和静态初始化块
  3. 将类的成员赋予初值(原始类型的成员的值为规定值,例如int型为0,float型为0.0f,boolean型为false;对象类型的初始值为null)
  4. 如果构造方法中存在this()调用(可以是其它带参数的this()调用)则执行之,执行完毕后进入第7步继续执行,如果没有this调用则进行下一步。(这个有可能存在递归调用其它的构造方法)
  5. 执行显式的super()调用(可以是其它带参数的super()调用)或者隐式的super()调用(缺省构造方法),此步骤又进入一个父类的构造过程并一直上推至Object对象的构造。
  6. 执行类申明中的成员赋值和初始化块。
  7. 执行构造方法中的其它语句。

其中第4步是比较麻烦的,因为this调用实际上会调用类的另外一个构造方法,最终应该是执行类的某个构造方法,它可能会显示的调用super,但是无论是否调用super,最终都是执行super的,也就是父类的构造方法并一直这样递归到Object,所以在子类和父类的构造中,首先构造或者说执行的是父类的构造,但是它是由子类的构造方法调用的,先于构造方法的方法体里面的内容,这个是由编译器决定的。所以我感觉简单直观一些的顺序表述应该是:

  1. 如果父类有静态成员赋值或者静态初始化块,执行静态成员赋值和静态初始化块
  2. 如果类有静态成员赋值或者静态初始化块,执行静态成员赋值和静态初始化块
  3. 将类的成员赋予初值(原始类型的成员的值为规定值,例如int型为0,float型为0.0f,boolean型为false;对象类型的初始值为null)
  4. 执行构造方法,并可能递归调用this(),最终先执行父类的构造方法并一直递归到Object的构造方法的执行
  5. 父类的构造方法执行完成后,执行类申明中的成员赋值和初始化块。
  6. 执行构造方法中的其它语句。

最终的简化顺序版本是:

  1. 父类的静态成员赋值和静态块
  2. 子类的静态成员和静态块
  3. 父类的构造方法
  4. 父类的成员赋值和初始化块
  5. 父类的构造方法中的其它语句
  6. 子类的成员赋值和初始化块
  7. 子类的构造方法中的其它语句

2006年11月16日更新:
针对留言中提到的那个文章中的问题发现这个顺序也是有不足的情况,这个顺序是一般的顺序,但是有可能被打破,留言中的那篇文章就是一个例子,因为在执行静态初始化块的时候先执行了类的构造,打破了这个一般顺序。所以这个顺序有个前提就是静态赋值和初始化块中没有对本类的实例化语句。
对于那个文章中的问题,作者最后的解决方法可行,但是不见得是最好的,可以简单的修改静态赋值和静态初始化块的顺序,修改后的代码片断为:
public class CachingEnumResolver {   
    private static Map CODE_MAP_CACHE;   
    /*MSGCODE->Category内存索引*/
    static {       
        CODE_MAP_CACHE = new HashMap();       
        //为了说明问题,我在这里初始化一条数据       
        CODE_MAP_CACHE.put("0","北京市");
    }
    //单态实例 一切问题皆由此行引起   
    private static final CachingEnumResolver SINGLE_ENUM_RESOLVER = new    CachingEnumResolver();   


作者: Cherami
原载: Java类的完整构造执行顺序
版权所有。转载时必须以链接形式注明作者和原始出处及本声明。 

http://www.niftyadmin.cn/n/610560.html

相关文章

通过代码快速上手C语言数据结构-栈与队列

3.1-3.4顺序栈 3.1初始化顺序栈 3.2顺序栈进栈运算 3.3顺序栈出栈运算 3.4顺序栈取栈顶元素运算 顺序栈头文件 #define TRUE 1 #define FALSE 0 #define Stack_Size 50/*顺序栈*/typedef struct {StackElementType elem[Stack_Size]; /*用来存放栈中元素的一维数组*/int …

【JokerのZYNQ7020】USER_TYPE_ETH_LINUX

软件环境:vivado 2017.4 硬件平台:XC7Z020 之前已经说了在linux下,怎么建立UDP、TCP-server、TCP-client的应用,而这些都是带有特定协议的以太网数据包,所以就好奇啦,有没有办法发送自定义type类型的…

C语言数据结构-树和二叉树-先序遍历-已知二叉树按照二叉链表方式存储,利用栈的基本操作写出先序遍历非递归形式的算法

先序遍历 已知二叉树按照二叉链表方式存储,利用栈的基本操作写出先序遍历非递归形式的算法: void pre_order(BiTree root);在遍历过程中,pre_order函数需要调用 visit_node 函数来实现对结点的访问,该函数声明如下: …

【JokerのZYNQ7020】PS_LWIP_POLL。

软件环境:vivado 2017.4 硬件平台:XC7Z020 说起zynq平台下PS端的lwip实现机制,有可能很多老哥跟我一样,具体并不十分清楚,我自己也是在很偶然的情况下看了一篇帖子才知道,居然利用的是PS端的dma收发…

C语言数据结构-树和二叉树-路径-假设二叉树采用二叉链表方式存储, root指向根结点,node 指向二叉树中的一个结点,编写函数 path,计算root到 node 之间的路径

路径 假设二叉树采用二叉链表方式存储, root指向根结点,node 指向二叉树中的一个结点,编写函数 path,计算root到 node 之间的路径,(该路径包括root结点和 node 结点)。path 函数声明如下&#…

C语言数据结构-树和二叉树-共同祖先 假设二叉树采用二叉链表方式存储, root指向根结点,p所指结点和q所指结点为二叉树中的两个结点,编写一个计算它们的最近的共同祖先

共同祖先 假设二叉树采用二叉链表方式存储, root指向根结点,p所指结点和q所指结点为二叉树中的两个结点,编写一个计算它们的最近的共同祖先,函数定义如下: BiTNode * nearest_ancestor(BiTree root, BiTNode *p, BiT…

【JokerのZYNQ7020】DNA_PORT。

软件环境:vivado 2017.4 硬件平台:XC7Z020 说起FPGA端最常用、最直接的加密手段,大概就是利用每个芯片独有的ID号 加密算法了,所以今天这篇,就来简单提一下如何获取7系列的每个芯片的DNA号。至于为什么说是7系…

C语言数据结构-树和二叉树-树转二叉树-使用队列,编写transfrom函数,将普通树转换成对应的二叉树

树转二叉树 使用队列,编写transfrom函数,将普通树转换成对应的二叉树。二叉树的相关定义如下: typedef int DataType;typedef struct Node{DataType data;struct Node* left;struct Node* right; }BiTNode, *BiTree;普通树节点的定义如下&a…