[Java] 类的初始化步骤

前言

类的初始化过程,可在多线程环境下进行的,为了简化,本文介绍的单线程情况下的类初始化步骤。

此外,继承情况下的顺序不是本文焦点,本文重点在于一个类的内部字段的初始化顺序。

"初始化" 一词,专门针对一个类而言。一个实例被创建过程中,实例字段被初始赋值,不称为初始化,而是直接称作实例的创建,以示区分。

正文

类的初始化,包括静态代码块的初始化、静态字段(类的字段)的初始化。

类的初始化触发条件:

1) T 是一个类,且 T 的一个实例被创建。

2) T 的一个静态方法被调用

3) T 的一个静态字段被赋值

4) T 的一个静态字段被使用,并且该字段不是一个字面常量。

5) T 是一级类,并且内嵌的断言语句被执行

本文的采用第 4) 种触发条件进行演示。

一个类的初始化步骤:

1. 在一个类被初始化之前,也就是在任何非字面类常量的类字段被初始化之前,字面类常量先完成初始化,如字段 static final Stirng a = "good"。

2. 当类被触发进行初始化,若其直接父类还没有被初始化,先对直接父类进行初始化;若直接父类的直接父类没有被初始化,则先对直接父类的直接父类进行初始化,以此类推,直到 Object 类或某一级别的祖父类已被初始化。

3. 初始化所有非类常量的类字段,同时初始化静态代码块,按文本序。

例子

Super, 父类,被初始化有输出

Test,测试主体类,继承 Super, 拥有四个字段:字面类常量 field1, 非字面的类常量 field2, 类变量 field3, 实例变量 field4; 拥有静态代码块,在类被初始化时被执行。

Test$InnerClass,Test 的内部类,在 Test 初始化时候被调用,用于表明字面类常量、非字面类常量两者被初始化的时间是不一样的。

InitDemo, 演示类的时候步骤。

具体代码

Super, 父类

public class Super {
    static {
        System.out.println("initializing Super ");
    }
}

Test, 测试的主体类

public class Test extends Super {

    public static final String field1 = "Test.field1";

    public static final String field2 = InnerClass.pint("Initializing Test.field2 ");
    public static String field3 = InnerClass.pint("Initializing Test.field3 ");

    public String field4 = InnerClass.pint("Initializing Test.field4 ");

    static{
        System.out.println("initializing Test class ");
        System.out.println("\t" + field1 + " - " + field2 + " - " + field3);
    }

    public Test(){
        System.out.println("in Test() ");
    }

    public void bMethod() {
        System.out.println("in Test.bMethod() ");
    }

    public static class InnerClass{
        public static String pint(String s){
            System.out.println(s);
            return s.substring(s.indexOf(" ") + 1);
        }

        static {
            System.out.println("initialzation in Test$innerClass ");
            System.out.println("\t" + field1 + " - " + field2 + " - " + field3);
        }
    }
}

InitDemo, 演示 Test 被初始化的步骤

public class InitDemo {
    public static void mian(){
        System.out.println(Test.field1);
        System.out.println("----------");
        System.out.println(Test.field2);
    }
}

输出如下

 1 Test.field1
 2 ----------
 3 initializing Super
 4 initialzation in Test$innerClass
 5     Test.field1 - null - null
 6 Initializing Test.field2
 7 Initializing Test.field3
 8 initializing Test class
 9     Test.field1 - Test.field2  - Test.field3
10 Test.field2 

从输出可以看出 :

在输出字面类常量 Test.field1 时候, Test 类并没有被初始化。而在输出非字面的类常量 Test.field2 时候,触发了 Test 的初始化。

在 Test 类被初始化之前,其父类 Super 先被初始化。

第 5 行是内部类 Test$innerClass 的输出,Test 类被初始化之前,非字面类字段 filed2、field3 都为 null,只有字面类常量 field1 已被赋值。

地 9 行是 Test 类静态代码块的初始化运行,此时类字段 field2、field3 均已被初始化。

类的初始化只是对类字段 ( field1、field2、field3 ) 进行初始化赋值,实例字段 field4 并不会被初始化赋值,需要等到创建实例是才会被初始化创建。

参考资料

12.4. Initialization of Classes and Interfaces, The Java Language Specification, Java SE 8 Edition

What is the best way to implement constants in Java?, stackOverflow

时间: 2024-10-17 19:32:36

[Java] 类的初始化步骤的相关文章

java类的初始化和构造函数

本人小白一枚,看java类的初始化的时候好晕的说,我觉着书上虽然说的对,但总觉得有些信息没说出来,没说清楚,看了好多文章博客的,现在有些感悟,来小写下总结,也算是为以后再次复习种个好果子. 先摘一下书上写的: 加载:将类的class文件读入内存,并为之创建一个java.lang.class对象. 连接:把类的二进制数据合并到JRE中,检查被加载的类是否有正确的内部结构,并和其他类协调一致.为类的静态FIELD分配内存,设置默认值,将类的二进制数据中的符号引用替换成直接引用. 初始化:主要对静态F

java类的初始化顺序

java类的初始化顺序 (2008-10-21 13:30:15) 转载▼ 标签: java 初始化 继承初始化 it 分类: Java 对于静态变量.静态初始化块.变量.初始化块.构造器,它们的初始化顺序依次是(静态变量.静态初始化块)>(变量.初始化块)>构造器.我们也可以通过下面的测试代码来验证这一点: public class InitialOrderTest { // 静态变量 public static String staticField = "静态变量";

Java类的初始化

Java提供了两种不同的初始化类型,分别是类的初始化和对象的初始化.类成员都是静态的,默认会设置一个值.对象的初始化会在构造函数里面进行.但如果想要赋给静态变量非默认值,或者是初始化一类共有的对象属性(不论调用哪个构造函数),那么就需要一些特殊的方法.提供了静态初始化块和非静态初始化块来处理这两种情况. 静态初始化块 静态初始化块是通过static{}来定义的.一个简单的代码示例如下: public class CorderStatic { staticint idx; static{ Syst

java类内容初始化顺序

在java类中一般有:成员变量.静态变量.成员方法.静态方法.构造方法.那么这几个的初始化顺序是什么呢? 初始化的先后顺序是:静态变量(类load进内存就初始化)------静态代码块(类load进内存就初始化)--------成员变量(对象初始化时)------------初始化块------------构造函数 测试程序如下: package com.evan; /* * 初始化顺序测试 */ public class InitialOrderTest { public static Str

java类的初始化和对象的创建顺序

学习java编程思想--类的初始化p146 类的加载顺序* 1加载器启动找到 xxx.class文件,通过extends关键字寻找基类,先加载基类* 2类初始化先初始化static成员变量和static--->* 2先初始化父类的static成员变量和static* 3再初始化本类的static成员变量和static * 类加载之后,对象创建开始* 1先加载父类的非静态成员变量(静态成员变量在类初始化的时候已经加载,非静态成员变量要随对象的创建而初始化)* 2先加载父类的构造函数* 3再加载本类

Java类的初始化过程及清理

一.类的数据成员初始化 Java中类的数据成员初试化可能有两种形式. 在定义类成员变量的地方直接提供初始化值(这是C++中不允许的) 在构造器中初试化.(Java中不存在类似C++中的初始化列表) 两者的区别是,第一种方式是数据成员直接初试化为提供的初始化值,而在构造器中初始化,其实在数据成员已经初试化为默认值(比如基本类型如int 初始化为0,引用类型初试为null,其实在在构造器已经不能称为初始化了,应该是赋值) 初始化顺序 如果有静态变量则先初始化静态变量,如果基类中也有静态变量且之前没有

Java类的初始化顺序 (静态变量、静态初始化块、变量、初始化块、构造器)(转)

大家在去参加面试的时候,经常会遇到这样的考题:给你两个类的代码,它们之间是继承的关系,每个类里只有构造器方法和一些变量,构造器里可能还有一段代码对变量值进行了某种运算,另外还有一些将变量值输出到控制台的代码,然后让我们判断输出的结果.这实际上是在考查我们对于继承情况下类的初始化顺序的了解. 我们大家都知道,对于静态变量.静态初始化块.变量.初始化块.构造器,它们的初始化顺序以此是(静态变量.静态初始化块)>(变量.初始化块)>构造器.我们也可以通过下面的测试代码来验证这一点: Java代码 p

java类的初始化块/执行顺序,实例化对象数据赋值

java里初始化一个类的对象,通过初始化快或者构造方法进行数据赋值.与其相关的执行代码有这么几种: 静态初始化块 初始化块 构造方法 静态初始化块 静态初始化块只在类加载时执行一次,同时静态初始化块只能给静态变量赋值,不能初始化普通的成员变量. 非静态初始化块 非静态初始化块在每次初始化实例对象的时候都执行一次,可以给任意变量赋值. 构造方法 在每次初始化实例对象时调用. 重点:执行顺序-> 在加载类时执行一次静态初始化块(之后不再调用). 在每次初始化实例对象时:先执行非静态初始化块,再执行构

java类的初始化过程

转自http://blog.csdn.net/monghuan/article/details/7404670 一个类可以使用不包含在任何方法体中的静态代码块,当类被载入时,静态代码块被执行,且只被执行一次,静态块常用来执行类属性的初始化. 一.类加载时,变量的初始化顺序:  1.首先,给静态成员变量分配内存空间,进行默认初始化   (整型为0,浮点型为0.0,布尔型为false,字符型为'\u0000',引用型为null)     2.其次,执行静态成员变量的初始化操作   --静态成员的初始