美文网首页
类加载机制

类加载机制

作者: 忘净空 | 来源:发表于2017-11-22 16:03 被阅读31次

    类加载的时机

    主动引用(有且只有下面的五种)

    1. 使用new关键字创建对象时,访问类的静态字段时;

    2. 使用java.lang.reflect包的方法进行反射调用的时候;

    3. 初始化一个类时,如果父类没有被初始化,则需要先初始化;

    4. 当虚拟机启动的时候,虚拟机先会初始化主类(main()方法所在的类)

    5. JDK1.7动态语言支持;

    被动引用(类不会加载)

    1. 子类引用父类的静态字段,不会导致子类的的初始化。

    2. 通过数组定义引用类。

    3. 访问类中的常量;

    类加载流程

    1. 加载

      1.1 加载二进制字节流到内存

      1.2 生成方法区数据结构

      1.3 生成java.lang.Class对象

    2. 连接

      2.1 验证

      • 文件格式验证: 版本号,是不是
      • 元数据验证:是否基础了final类?是不是有父类?是不是实现了抽象类的方法?
      • 字节码验证:跳转指令是否合法?
      • 字符引用验证:引用的类是否存在?类及接口是否有访问权限?

      2.2 准备
      类变量(即静态变量而不是实例变量)分配内存,并为类设置初始值(方法区)

      • public static int v=1;
      • 在准备阶段中个,v会被设置为0
      • 在初始化的<clinit>中才会被设置为1
      • 对于static final类型,在准备阶段就会被赋上准确的值
      • public static final int v=1

      2.3 解析

      • 字符引用转换为直接引用:com.XXX.sudent这个就是个符号,直接引用就是它在内存的地址
    3. 初始化
      执行类构造器<clinit>方法,<clinit>方法由静态代码块、静态变量构成和赋值语句构成。

      • <clinit>方法线程安全: 基于它可以实现线程安全单例,参考单例模式
      • <clinit>方法执行时,发现父类的没有执行先执行父类的<clinit>方法
    4. 使用

    5. 卸载

    类加载器

    类加载器分类

    1. 启动类加载器

    2. 扩展加载器

    3. 应用类加载器

    4. 自定义类加载器

    备注:类加载器之间不是继承的关系,而是组合的关系。

    双亲委派模式

    类加载器:字底向上查找是否已经加载类,自顶向下加载类。

    双亲委派模式:为了维护类加载的安全。

    破坏双亲委派模式

    1. 接口在rt.jar实现在应用程序,启动类加载器如何加载应用类加载器的类,例如:SPI机制

      线程上线文类加载器可以解决这个问题,它模式的类加载器是应用类加载器。

    2. 热替换

    问题 1:

    static{
        a = 300;
        System.out.println(a);//这里会报错,声明语句之前只能赋值不能引用
    }
    public static int a = 10;
    
    

    问题 2:

    类加载中只用类变量及静态代码执行,那么构造方法、普通代码块啥时候执行呢,其实这个不属于类加载,它属于类的实例化,在类的实例化是执行。

    public class Line {  
        static {  
            System.out.println("静态代码块执行:loading line");  
        }  
      
        public static String s = getString();  
      
        private static String getString() {  
            System.out.println("给静态变量赋值的静态方法执行:loading line");  
            return "ss";  
        }  
      
        public static void test() {  
            System.out.println("普通静态方法执行:loading line");  
        }  
      
        public Line() {  
            System.out.println("构造方法执行:loading line");  
        }  
      
        {  
            System.out.println("构造代码块执行");  
        }  
    }
    public class CodeBlockTest {  
        public static void main(String[] args) {  
            System.out.println("主方法");  
            {  
                System.out.println("main方法中最开始的,普通代码块执行");  
            }  
            Line line = new Line();  
            System.out.println("...............");  
            Line line1 = new Line();  
            System.out.println("...............");  
            {  
                System.out.println("main方法中结尾事的,普通代码块执行");  
            }  
        }  
    }
    

    结果大家自己执行,注意下:构造代码块在创建对象时被调用,每次创建对象都会被调用,并且构造代码块的执行次序优先于类构造函数。

    问题 3:

    public class Singleton {
        private static Singleton mInstance = new Singleton();// 位置1
        public static int counter1;
        public static int counter2 = 0;
    //        private static Singleton mInstance = new Singleton();// 位置2
        private Singleton() {
            counter1++;
            counter2++;
        }
    
        public static Singleton getInstance() {
            return mInstance;
        }
    }
    
    class InitDemo {
        public static void main(String[] args) {
            Singleton singleton = Singleton.getInstance();
            System.out.println("counter1: " + singleton.counter1);
            System.out.println("counter2: " + singleton.counter2);
        }
    }
    位置1:
    counter1=1
    counter2=0
    
    位置2:
    counter1=1
    counter2=1
    

    相关文章

      网友评论

          本文标题:类加载机制

          本文链接:https://www.haomeiwen.com/subject/reqovxtx.html