Java类加载的时机

对于类加载过程中初始化阶段,虚拟机严格规定了有且只有四种情况必须立即对类进行“初始化”:

  1. 遇到new、getstatic、putstatic和invokestatic这4条指令码时,如果类没有初始化,则需要先进行初始化。这4条指令对应的java场景是:
    1. 使用new关键字实例化对象
    2. 读取或设置一个静态字段,不包括被final修饰的、已在编译器把结果放入常量池的静态字段
    3. 调用一个静态方法
  2. 使用java.lang.reflect包方法对类进行发射调用
  3. 当初始化一个类的时候,发现其父类还没有进行过初始化,则需要先初始化其父类。但是对于一个接口来说,只有在用到其父接口的时候才会去初始化。
  4. 当虚拟机启动时,用户需要指定一个要执行的类,虚拟机会先初始化这个类

以上这四种情况被称为是对一个类的主动引用。除此之外所有引用类的方式,都不会触发初始化,被称为被动应用。

下面是几个被动引用的例子,假设

[java] view plaincopyprint?
class SuperClass{  
    public static int value = 13;  
 
    static {  
        System.out.println("SuperClass Init");  
    }  
}  
 
class SubClass extends SuperClass{  
    static {  
       System.out.println("SubClass Init");  
   }  
}  

 

 

  1. 通过子类引用父类的静态字段,不会导致子类初始化
public class ClassLoadTest {  
    public static void main(String[] args) {  
       System.out.println(SubClass.value);  
   }  
}  


 

#=>SuperClass Init

#=>13

对于静态字段,只有直接定义这个字段的类才会被初始化。

  1. 通过数组来定义引用类,不会触发此类的实例化

 

public class ClassLoadTest {  
    public static void main(String[] args) {  
        SuperClass[] superClasses = new SuperClass[10];  
    }  
}  
  1. 常量在编译阶段会存入调用类的常量池,本质上没有直接引用到定义常量的类,因此不会触发定义常量的类的初始化。
    class SuperClass{  
        public static final int value = 13;  
      
        static {  
            System.out.println("SuperClass Init");  
        }  
    }  
      
    public class ClassLoadTest {  
        public static void main(String[] args) {  
            System.out.print(SuperClass.value);  
        }  
    } 

 

在编译阶段SuperClass.value的值就被存在了ClassLoadTest的常量池中了,对SuperClass.value的引用实际上相当于ClassLoadTest内部的引用,所以在ClassLoadTest的类文件中并没有SuperClass类的符号引用入口,这两个类在编译后没有任何关系了。



 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值