<clinit> 与 <init> 方法

<clinit>方法

先理解 类初始化阶段 的含义: 该阶段负责为类变量赋予正确的初始值, 是一个类或接口被首次使用前的最后一项工作

  • <clinit>方法 的执行时期: 类初始化阶段(该方法只能被jvm调用, 专门承担类变量的初始化工作)
  • <clinit>方法 的内容: 所有的类变量初始化语句和类型的静态初始化器
  • 类的初始化时机: 即在java代码中首次主动使用的时候, 包含以下情形:
    - (首次)创建某个类的新实例时–new, 反射, 克隆 或 反序列化;
    - (首次)调用某个类的静态方法时;
    - (首次)使用某个类或接口的静态字段或对该字段(final 字段除外)赋值时;
    - (首次)调用java的某些反射方法时;
    - (首次)初始化某个类的子类时;
    - (首次)在虚拟机启动时某个含有 main() 方法的那个启动类

注意: 并非所有的类都会拥有一个**<clinit>**方法, 满足下列条件之一的类不会拥有方法:

  1. 该类既没有声明任何类变量,也没有静态初始化语句;
  2. 该类声明了类变量,但没有明确使用类变量初始化语句或静态初始化语句初始化;
  3. 该类仅包含静态 final 变量的类变量初始化语句,并且类变量初始化语句是编译时常量表达式;
  • 案例解析

1.关于编译错误illegal forward reference(违法向前引用):

package com.jvm.exercises;

/**
 * @author dimdark
 */
public class ClinitAndInitTest {

    static ClinitAndInitTest test = new ClinitAndInitTest();

    // 静态语句块
    static {
        System.out.println("static statements block");
        // 注意 test 与 s 的声明位置
        System.out.println(test); // 调用类变量test, 未出现编译错误
        System.out.println(s);    // 调用类变量s, 出现编译错误illegal forward reference
    }

    static String s = "string";

}

结论:
在static语句块中使用到静态变量时一定要将该静态变量的声明语句放在static语句块的前面, 否则会发生illegal forward references的编译错误

2.关于静态常量(static final类型)的赋值时机所引起的问题:
// 对比下面两段代码的输出结果

package com.jvm.exercises;

/**
 * @author dimdark
 */
public class ClinitTestFive {

    private static ClinitTestFive test;

    static {
        test = new ClinitTestFive();
    }

    private static final String name = "string_name";

    private String testName;

    private ClinitTestFive() {
        testName = name;
    }

    public static void main(String[] args) {
        System.out.println(test.testName); // 输出结果为: string_name
    }

}

package com.jvm.exercises;

/**
 * @author dimdark
 */
public class ClinitTestFive {

    private static ClinitTestFive test;

    static {
        test = new ClinitTestFive();
    }

    private static final String name = new String("string_name"); 

    private String testName;

    private ClinitTestFive() {
        testName = name;
    }

    public static void main(String[] args) {
        System.out.println(test.testName); // 输出结果为: null
    }

}

分析: 上述代码段1中由于name被赋予字符串字面量"string_name", 故在name声明时其值就是"string_name"; 而代码段2中由于使用new String方式为name赋值, 导致name在声明时未被初始化(默认为null), 直到static语句块执行后才会被初始化为"string_name", 而static语句块执行期间调用类的构造方法, 构造方法中使用了name, 注意此时name并未被赋值,因此testName为null.

结论: 要保证静态常量在使用前被赋予值, 否则会出现意想不到的情况.

<init>方法:

  • <init>方法 的执行时期: 对象的初始化阶段
  • 实例化一个类的四种途径:
    1.调用 new 操作符
    2.调用 Class 或 java.lang.reflect.Constructor 对象的newInstance()方法
    3.调用任何现有对象的clone()方法
    4.通过 java.io.ObjectInputStream 类的 getObject() 方法反序列化
  • 小案例:
package com.jvm.exercises;


/**
 * @author dimdark
 */
public class InitTest {

    private int code = 0;

    InitTest() {
        code = 1;
        name = "init_name";
    }

    private String name = "name";

    @Override
    public String toString() {
        return "InitTest{" +
                "code=" + code +
                ", name='" + name + '\'' +
                '}';
    }

    public static void main(String[] args) {
        System.out.println(new InitTest()); // InitTest{code=1, name='init_name'}
    }

}
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值