一 问题
使用 static + final 修饰的字段的显式赋值的操作,到底是在哪个阶段进行的赋值?
-
情况 1:在链接阶段的准备环节赋值
-
情况 2:在初始化阶段<clinit>()中赋值
二 结论
赋值时机见如下结论。
-
对于基本数据类型的字段来说,如果使用 static final 修饰,则显式赋值(直接赋值常量,而非调用方法)通常是在链接阶段的准备环节进行。
-
对于 String 来说,如果使用字面量的方式赋值,使用 static final 修饰的话,则显式赋值通常是在链接阶段的准备环节进行。
-
在初始化阶段<clinit>()中赋值的情况: 排除上述的在准备环节赋值的情况之外的情况。
三 最终结论
使用 static + final 修饰,且显示赋值中不涉及到方法或构造器调用的基本数据类型或 String 类型的显式赋值,是在链接阶段的准备环节进行。
四 实战——哪些场景下 Java 编译器不会生成 clinit 方法
1 代码
/**
* 哪些场景下,java编译器就不会生成<clinit>()方法
*/
public class InitializationTest1 {
// 场景1:对应非静态的字段,不管是否进行了显式赋值,都不会生成<clinit>()方法
public int num = 1;
// 场景2:静态的字段,没有显式的赋值,不会生成<clinit>()方法
public static int num1;
// 场景3:比如对于声明为 static final 的基本数据类型的字段,不管是否进行了显式赋值,都不会生成<clinit>()方法
public static final int num2 = 1;
// 不加 final 的变量会生成 clinit
public static int num3 = 1;
}
2 图解
五 实战——到底赋值在哪个阶段进行
1 代码
package chapter03.java;
import java.util.Random;
/**
*
* 一 问题
* 使用 static + final 修饰的字段的显式赋值的操作,到底是在哪个阶段进行的赋值?
* 情况1:在链接阶段的准备环节赋值
* 情况2:在初始化阶段<clinit>()中赋值
*
* 二 结论
* 在链接阶段的准备环节赋值的情况:
* 1. 对于基本数据类型的字段来说,如果使用 static final 修饰,则显式赋值(直接赋值常量,而非调用方法)通常是在链接阶段的准备环节进行
* 2. 对于 String 来说,如果使用字面量的方式赋值,使用 static final 修饰的话,则显式赋值通常是在链接阶段的准备环节进行
*
* 在初始化阶段<clinit>()中赋值的情况:
* 排除上述的在准备环节赋值的情况之外的情况。
*
* 三 最终结论、
* 使用 static + final 修饰,且显示赋值中不涉及到方法或构造器调用的基本数据类型或String类型的显式赋值,是在链接阶段的准备环节进行。
*/
public class InitializationTest2 {
// 在链接阶段的准备环节赋值
public static final int INT_CONSTANT = 10;
public static final String s0 = "helloworld0";
public static final int NUM = 2;
// 在初始化阶段<clinit>()中赋值
public static String s2 = "helloworld2";
public static int a = 1;
public static final Integer INTEGER_CONSTANT1 = Integer.valueOf(100);
public static Integer INTEGER_CONSTANT2 = Integer.valueOf(1000);
public static final String s1 = new String("helloworld1");
public static final int NUM1 = new Random().nextInt(10);
}