• 类加载器


      1.类加载过程

      1.1 加载过程:将.class文件通过IO流的方式加载到内存当中   

        1.将.class文件字节码内容加载到内存当中
        2.先会将静态数据转换成方法区中的运行的数据结构
        3.在堆内存当中生成一个代表这个类的Class对象,这个Class类的对象就是作为方法区数据访问的入口 Class.forName(com.wdksoft.User);

        加载指的是将类的class文件读入到内存,并为之创建一个java.lang.Class对象,也就是说,当程序中使用任何类时,系统都会为之建立一个java.lang.Class对象。

       类的加载由类加载器完成,类加载器通常由JVM提供,这些类加载器也是前面所有程序运行的基础,JVM提供的这些类加载器通常被称为系统类加载器。除此之外,开发者可以通过继承ClassLoader基类来创建自己的类加载器。

         1.2 链接

       当类被加载之后,系统为之生成一个对应的Class对象,接着将会进入连接阶段,连接阶段负责把类的二进制数据合并到JRE中。类连接又可分为如下3个阶段。

        1.验证阶段:验证字节码文件的准确性,包含文件格式,元数据,符号引用,字节码等等
        2.准备阶段:给类中的静态变量分配内存,并赋予初始值
        3.解析阶段:将虚拟机常量池的符号引用替换成字节引用的过程
       初始化过程:初始化过程就会对类中的静态变量初始化为指定的值,执行静态代码块,执行构造器

      2.类加载器种类     

        2.1 引导(跟)类加载器:负责加载JRE核心类库,jre包下rt.jar,charsets.jar等,C++语言编写,无法直接访问,所以结果为null 

         //虚拟机内置的三个类加载器
            //1.根类加载器(没有父加载器)
            ClassLoader classLoader = Object.class.getClassLoader();
            System.out.println("[跟类加载器]object类 的类加载器是:"+classLoader);

        控制台打印结果

       

        2.2 扩展类加载器ExtClassLoader:负责加载JRE扩展目录ext中的jar包   

    //2.扩展类加载器(扩展类加载器的父加载器是跟类加载器)
            ClassLoader classLoader1 = DNSNameService.class.getClassLoader();
            System.out.println("[扩展类加载器]DNSNameService的类加载器是:"+classLoader1);

        控制台打印结果


        2.3 系统类加载器AppClassLoader:负责加载classPath路径下的类包

    //3.系统类加载器(系统类加载器的父加载器是扩展类加载器)
            //自己写的类默认走系统类加载器
            ClassLoader classLoader2 = ClassLoader1.class.getClassLoader();
            System.out.println("[系统类加载器]ClassLoader1的类加载器为:"+classLoader2);

        控制台打印结果


        2.4 自定义类记载器:负责加载用户自定义下的类包
      实现自定义类加载器:extends ClassLoader,重写loadClass方法进行类的加载

      3.类加载机制

      3.1 全盘负责委托机制
        当进行类加载的时候,如果手动指定了ClassLoader,那么该类所依赖和引用的类也由这个类加载器进行加载
        User->UserParent
        指定User使用特定的类加载器,那么跟User类有依赖和引用关系的类也用这个类加载器进行加载

      3.2 双亲委派机制

      

         目的:保证字节码文件只被加载一次

        指先委托父类加载器寻找目标类,如果父类加载器无法进行类的加载则子类加载器自身处理
        3.2.1 沙箱安全机制:自定义的String.class不会被加载,这样可以防止核心API库被随意篡改

        3.2.2 避免类重复加载:当附加在其加载了该类是,就没有必要子类加载器也进行加载

     

       4. 如何破坏双亲委派机制

        4.1 重写ClassLoad类中的loadClass方法,指定加载哪一个类

        4.2 手动调用系统类加载器Thread.currentThread().getContextClassLoader();

        4.3 重写findClass

      

      5. 监控类加载过程

      在当前启动类当中加入-verbose:class参数,启动则可以看到整个类加载的过程

      6. 热部署

        时时检测类,如果类发生更改则自动进行重新编译,编译之后重新加载该类,提高开发的效率,修改代码后无需重启应用

  • 相关阅读:
    【Java】【IDE】【Jetbrain Idea】Intellij IDEA 快捷键整理
    【Linux】【Services】【KVM】virsh命令详解
    【Linux】【Services】【KVM】安装与简单配置
    【Linux】【Services】【Docker】Docker File
    【Linux】【Services】【Docker】网络
    【Linux】【Services】【Docker】应用
    【Linux】【Services】【Docker】基础理论
    【Python】【Module】json and pickle
    【Python】【Module】hashlib
    Highcharts 对数组的要求
  • 原文地址:https://www.cnblogs.com/szhhhh/p/12469977.html
Copyright © 2020-2023  润新知