Java_类的加载

基本说明:

反射机制是java实现动态语言的关键,也就是通过反射实现类动态加载

1.静态加载:编译时加载相关的类,如果没有则报错,依赖性太强

2.动态加载:运行时加载需要的类,如果运行时不用该类,即使不存在该类,也不会报错,降低了依赖性

类加载时机:

1.当创建对象时(new)

2.当子类被加载时

3.调用类中的静态成员时

4.通过反射

加载阶段:

JVM在该阶段的主要目的是将字节码从不同的数据源(可能是class文件,也可能是jar包,甚至网络)转化为二进制字节流加载到内存中,并生成一个代表该类的java.lang.Class对象

连接阶段-验证:

1.目的是确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全

2.包括:文件格式验证(是否以魔数 oxcafebabe开头),元数据验证,字节码验证和符号引用验证

3.可以考虑使用 -Xverify:none 参数来关闭大部分的类验证措施,缩短虚拟机类加载的时间

连接阶段-准备:

1.JVM会在该阶段对静态变量分配内存并默认初始化(对应数据类型的默认初始值,如0,0L,null,false等)这些变量所使用的内存都将在方法区中进行分配

java 复制代码
class A {
    //属性-成员变量-字段
    //分析类加载的链接阶段-准备 属性是如何处理
    //1. n1 是实例属性, 不是静态变量,因此在准备阶段,是不会分配内存
    //2. n2 是静态变量,分配内存 n2 是默认初始化 0 ,而不是20
    //3. n3 是static final 是常量, 他和静态变量不一样, 因为一旦赋值就不变 n3 = 30
    public int n1 = 10;
    public static  int n2 = 20;
    public static final  int n3 = 30;
}

连接阶段-解析:

1.虚拟机将常量池内的符号引用替换为直接引用的过程

Initialization(初始化)

1.到初始化阶段,才真正开始执行类中定义的Java程序代码,此阶段是执行<clinit>()方法的过程

2.<clinit>()方法是由编译器按语句在源文件中出现的顺序,依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并进行合并

3.虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确的加锁,同步,如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的<clinit>()方法,其他线程都需要阻塞等待,直到活动线程执行<clinit>()方法完毕

java 复制代码
package com.reflection.ClassLoad;

//演示类加载-初始化阶段
public class ClassLoad_ {
    public static void main(String[] args) {
        //分析:
        //1.加载B类,并生成B的一个Class对象
        //2.连接 num = 0;
        //3.初始化阶段
        //依次自动收集类中的所有静态变量的赋值动作和静态代码块中的语句,并合并
        /*
        clinit(){
            System.out.println("B 的静态代码块被执行.....");
            num = 300;
            static int num = 100;
        }
        合并:num = 100
         */
        //如果直接使用类的静态属性/变量也会导致类的加载
        System.out.println(B.num);
        //输出 B 的静态代码块被执行..... 100

        B b = new B();
        System.out.println(B.num);
        //输出 B 的静态代码块被执行..... 100 B 的构造器被执行..... 100

        //虚拟机会保证一个类的<clinit>()方法在多线程环境中被正确的加锁,同步,
        //如果多个线程同时去初始化一个类,那么只会有一个线程去执行这个类的<clinit>()方法,
        //其他线程都需要阻塞等待,直到活动线程执行<clinit>()方法完毕
        //正因为有这个机制,才能保证某个类在内存中只有一个class对象
    }
}
class B{
    static {
        System.out.println("B 的静态代码块被执行.....");
        num = 300;
    }
    static int num = 100;

    public B(){
        System.out.println("B 的构造器被执行.....");
    }
}
相关推荐
Javatutouhouduan3 小时前
2026Java面试的正确打开方式!
java·高并发·java面试·java面试题·后端开发·java编程·java八股文
chao1898444 小时前
基于 SPEA2 的多目标优化算法 MATLAB 实现
开发语言·算法·matlab
JAVA面经实录9174 小时前
Java初级最终完整版学习路线图
java·spring·eclipse·maven
赏金术士4 小时前
Kotlin 习题集 · 高级篇
android·开发语言·kotlin
Cat_Rocky5 小时前
k8s-持久化存储,粗浅学习
java·学习·kubernetes
楼兰公子5 小时前
buildroot 在编译rust时裁剪平台类型数量的方法
开发语言·后端·rust
知识领航员5 小时前
蘑兔AI音乐深度实测:功能拆解、实测表现与适用场景
java·c语言·c++·人工智能·python·算法·github
吴声子夜歌5 小时前
Go——并发编程
开发语言·后端·golang
释怀°Believe5 小时前
Spring解析
java·后端·spring
ooseabiscuit6 小时前
Laravel4.x:现代PHP框架的奠基之作
java·开发语言·php