反射-分析类的初始化

什么时候会发生类初始化?
  类的主动引用(一定会发生类的初始化)
    当虚拟机启动,先初始化main方法所在的类new一个类的对象
    调用类的静态成员(除了final常量)和静态方法
    使用java.lang.reflect包的方法对类进行反射调用
    当初始化一个类,如果其父类没有被初始化,则先会初始化它的父类
  类的被动引用(不会发生类的初始化)
    当访问一个静态域时,只有真正声明这个域的类才会被初始化。如:当通过子类引用父类的静态变量,不会导致子类初始化
    通过数组定义类引用,不会触发此类的初始化
    引用常量不会触发此类的初始化(常量在链接阶段就存入调用类的常量池中了)

 

package 反射;//测试类什么时候会初始化

public class Main {
    static {
        System.out.println("Main类被加载");
    }
    public static void main(String[] args)throws ClassNotFoundException {
    //主动引用,new的时候会主动引用
    Son son=new Son();//注:由于new子类的时候没有另外new父类,所以父类并不没有被初始化,所以在初始化子类的时候,会先初始化父类
    
    //反射也会产生主动引用
   Class.forName("反射.Son");//因为我们不知道他有什么东西,只知道他的包名,所以我们直接丢一个包名进去,正常情况下写包名全路径,不要像这个只写类名

    //不会产生类的引用的方法
    System.out.println(Son.b);//通过子类去调用父类的属性写方法,虽然是通过子类去调用的,但是不会影响到子类,子类也不会产生初始化
    Son[] array=new Son[5];//那这个代码只是一个名字和一个空间所以不会被加载,只有Main类的方法被加载,因为虚拟机启动后,main类就会被加载
    System.out.println(Son.M);//常量不会引起父类跟子类的初始化,因为常量在链接阶段就已经调入了类的常量数据池


    }
}
class lather{
    static int b=2;
    static {
        System.out.println("父类被加载");
    }
}
class Son extends lather{
    static {
        System.out.println("子类被加载");
        m=300;
    }
    static int m=100;
    static final int M=1;
}

 

posted @ 2022-06-07 14:24  hollg  阅读(56)  评论(0)    收藏  举报