这次绝对让你深刻理解java类的加载机制

609 阅读24分钟

前言  

是否真的理解java的类加载机制?点进文章的盆友不如先来做一道非常常见的面试题,如果你能做出来,可能你早已掌握并理解了java的类加载机制,若结果出乎你的意料,那就很有必要来了解了解java的类加载机制了。代码如下 

 package com.jvm.classloader;

class Father2{
    public static String strFather="HelloJVM_Father";

    static{
        System.out.println("Father静态代码块");
    }
}

class Son2 extends Father2{
    public static String strSon="HelloJVM_Son";

    static{
        System.out.println("Son静态代码块");
    }
}

public class InitativeUseTest2 {
    public static void main(String[] args) {

       System.out.println(Son2.strSon);
    }
}

运行结果: 

 Father静态代码块
        Son静态代码块
        HelloJVM_Son

嗯哼?其实上面程序并不是关键,可能真的难不倒各位,不妨做下面一道面试题可好?如果下面这道面试题都做对了,那没错了,这篇文章你就不用看了,真的。 

 package com.jvm.classloader;

class YeYe{
    static {
        System.out.println("YeYe静态代码块");
    }
}

class Father extends YeYe{
    public static String strFather="HelloJVM_Father";

    static{
        System.out.println("Father静态代码块");
    }
}

class Son extends Father{
    public static String strSon="HelloJVM_Son";

    static{
        System.out.println("Son静态代码块");
    }
}

public class InitiativeUse {
    public static void main(String[] args) {
        System.out.println(Son.strFather); 
    }
}

各位先用“毕生所学”来猜想一下运行的结果是啥… 

注意了… 

注意了… 

注意了… 

运行结果:

YeYe静态代码块
    Father静态代码块
    HelloJVM_Father

是对是错已经有个数了吧,我就不拆穿各位的小心思了… 

以上的面试题其实就是典型的java类的加载问题,如果你对Java加载机制不理解,那么你可能就错了上面两道题目的。这篇文章将通过对Java类加载机制的讲解,让各位熟练理解java类的加载机制。 

其实小瓜哥还是想在给出一道题,毕竟各位都已经有了前面两道题的基础了,那么请看代码: 

package com.jvm.classloader;

class YeYe{
    static {
        System.out.println("YeYe静态代码块");
    }
}

class Father extends YeYe{
    public final static String strFather="HelloJVM_Father";

    static{
        System.out.println("Father静态代码块");
    }
}

class Son extends Father{
    public static String strSon="HelloJVM_Son";

    static{
        System.out.println("Son静态代码块");
    }
}

public class InitiativeUse {
    public static void main(String[] args) {
        System.out.println(Son.strFather);
    }
}

注意了 

注意了 

注意了 

运行结果:HelloJVM_Father 

1 冲动的小白童鞋看到了运行结果,果断的注销了博客账户… 

1、什么是类的加载 

JVM重要的一个领域:类加载 

而类加载必然涉及类加载器,下面我们先来了解一下类的加载。 

类加载: 
1、在java代码中,类型的加载、连接、与初始化过程都是在程序运行期间完成的(类从磁盘加载到内存中经历的三个阶段)【牢牢记在心里】 
2、提供了更大的灵活性,增加了更多的可能性 

虽然上面的第一句话非常简短,但是蕴含的知识量却是巨大的!包含两个重要的概念: 

1、类型 

定义的类、接口或者枚举称为类型而不涉及对象,在类加载的过程中,是一个创建对象之前的一些信息 

2、程序运行期间 

程序运行期间完成典型例子就是动态代理,其实很多语言都是在编译期就完成了加载,也正因为这个特性给Java程序提供了更大的灵活性,增加了更多的可能性 

类的加载指的是将类的.class文件中的二进制数据读入到内存中,将其放在运行时数据区的方法区内,然后在堆区创建一个 java.lang.Class对象(JVM规范并未说明Class对象位于哪里,HotSpot虚拟机将其放在方法区中),用来封装类在方法区内的数据结构。类的加载的最终产品是位于堆区中的 Class对象, Class对象封装了类在方法区内的数据结构,并且向Java程序员提供了访问方法区内的数据结构的接口。 

Class对象是存放在堆区的,不是方法区,这点很多人容易犯错。类的元数据才是存在方法区的。【元数据并不是类的Class对象。Class对象是加载的最终产品,类的方法代码,变量名,方法名,访问权限,返回值等等都是在方法区的】 

JDK7创建Class实例存在堆中;因为JDK7中JavaObjectsInPerm参数值固定为false。 
JDK8移除了永久代,转而使用元空间来实现方法区,创建的Class实例依旧在java heap(堆)中 


编写一个新的java类时,JVM就会帮我们编译成class对象,存放在同名的.class文件中。在运行时,当需要生成这个类的对象,JVM就会检查此类是否已经装载内存中。若是没有装载,则把.class文件装入到内存中。若是装载,则根据class文件生成实例对象。 

怎么理解Class对象与new出来的对象之间的关系呢? 

new出来的对象以car为例。可以把car的Class类看成具体的一个人,而new car则是人物映像,具体的一个人(Class)是唯一的,人物映像(new car)是多个的。镜子中的每个人物映像都是根据具体的人映造出来的,也就是说每个new出来的对象都是以Class类为模板参照出来的!为啥可以参照捏?因为Class对象提供了访问方法区内的数据结构的接口哇,上面提及过了喔! 

算了参照下面这张图理解吧,理解是其次,重点是话说这女孩子蛮好看的。 


总结: 

类的加载简单来说就是:

.class文件(二进制数据)——>读取到内存——>数据放进方法区——>堆中创建对应Class对象——>并提供访问方法区的接口 

1.1类加载注意事项 

1、类加载器并不需要等到某个类被 “首次主动使用” 时再加载它关于首次主动使用这个重要概念下文将讲解 

2、JVM规范允许类加载器在预料某个类将要被使用时就预先加载它 

3、如果在预先加载的过程中遇到了.class文件缺失或存在错误,类加载器必须在程序首次主动使用该类时才报告错误(LinkageError错误)如果这个类一直没有被程序主动使用,那么类加载器就不会报告错误。 

4、类的加载的最终产品是位于堆区中的 Class对象 

1.2.加载.calss文件的方式 

(1)从本地系统中直接加载 

(2)通过网络下载.class文件 

(3)从zip,jar等归档文件中加载.class文件 

(4)从专用数据库中提取.class文件 

(5)将java源文件动态编译为.class文件 

首先给各位打个预防针:可能没有了解过JVM的童鞋可能看的很蒙,感觉全是理论的感觉,不勉强一字一句的“死看”,只要达到一种概念印象就好!毕竟学习是一种循进渐进的过程。

2、类的生命周期 

从上图可知,类从被加载到虚拟机内存开始,到卸载出内存为止,它的整个生命周期包括 7 个阶段,而验证、准备、解析 3 个阶段统称为连接。 

加载、验证、准备、初始化和卸载这 5 个阶段的顺序是固定确定的,类的加载过程必须按照这种顺序开始(注意是“开始”,而不是“进行”),而解析阶段则不一定:它在某些情况下可以在初始化后再开始,这是为了支持 Java 语言的运行时绑定【也就是java的动态绑定/晚期绑定】。 

2、1.加载 

  • 在上一节中已经详细提到过类的加载过程,类加载就是.class文件(二进制数据)——>读取到内存——>数据放进方法区——>堆中创建对应Class对象——>并提供访问方法区的接口 
  • 相对于类加载的其他阶段而言,加载阶段(准确地说,是加载阶段获取类的二进制字节流的动作)是可控性最强的阶段,因为开发人员既可以使用系统提供的类加载器来完成加载,也可以自定义自己的类加载器来完成加载。 
  • 加载阶段完成后,虚拟机外部的二进制字节流就按照虚拟机所需的格式存储在方法区之中,而且在Java堆中也创建一个 java.lang.Class类的对象,这样便可以通过该对象访问方法区中的这些数据。 

2、2.验证 

验证:确保被加载的类的正确性。 关于验证大可不必深入但是了解类加载机制必须要知道有这么个过程以及知道验证就是为了验证确保Class文件的字节流中包含的信息符合当前虚拟机的要求即可。 所以下面关于验证的内容作为了解即可! 

验证是连接阶段的第一阶段,这一阶段的目的是为了确保Class文件的字节流中包含的信息符合当前虚拟机的要求,并且不会危害虚拟机自身的安全。验证阶段大致会完成4个阶段的检验动作: 

文件格式验证:验证字节流是否符合Class文件格式的规范;例如:是否以 0xCAFEBABE开头、主次版本号是否在当前虚拟机的处理范围之内、常量池中的常量是否有不被支持的类型。 

元数据验证:对字节码描述的信息进行语义分析(注意:对比javac编译阶段的语义分析),以保证其描述的信息符合Java语言规范的要求;例如:这个类是否有父类,除了java.lang.Object之外。 

字节码验证:通过数据流和控制流分析,确定程序语义是合法的、符合逻辑的。 

符号引用验证:确保解析动作能正确执行。 

验证阶段是非常重要的,但不是必须的,它对程序运行期没有影响,如果所引用的类经过反复验证,那么可以考虑采用 -Xverifynone参数来关闭大部分的类验证措施,以缩短虚拟机类加载的时间。 

2、3.准备【重点】 

当完成字节码文件的校验之后,JVM 便会开始为类变量分配内存并初始化。准备阶段是正式为类变量分配内存并设置类变量初始值的阶段,这些内存都将在方法区中分配。 

这里需要注意两个关键点,即内存分配的对象以及初始化的类型。 

内存分配的对象:要明白首先要知道Java 中的变量有类变量以及类成员变量两种类型,类变量指的是被 static 修饰的变量,而其他所有类型的变量都属于类成员变量。在准备阶段,JVM 只会为类变量分配内存,而不会为类成员变量分配内存。类成员变量的内存分配需要等到初始化阶段才开始(初始化阶段下面会讲到)。 

举个例子:例如下面的代码在准备阶段,只会为 LeiBianLiang属性分配内存,而不会为 ChenYuanBL属性分配内存。 

public static int LeiBianLiang = 666;
public String ChenYuanBL = "jvm";

初始化的类型:在准备阶段,JVM 会为类变量分配内存,并为其初始化(JVM 只会为类变量分配内存,而不会为类成员变量分配内存,类成员变量自然这个时候也不能被初始化)。但是这里的初始化指的是为变量赋予 Java 语言中该数据类型的默认值,而不是用户代码里初始化的值。 

例如下面的代码在准备阶段之后,LeiBianLiang 的值将是 0,而不是 666。 

public static int LeiBianLiang = 666;

注意了!!! 

注意了!!! 

注意了!!! 

但如果一个变量是常量(被 static final 修饰)的话,那么在准备阶段,属性便会被赋予用户希望的值。例如下面的代码在准备阶段之后,ChangLiang的值将是 666,而不再会是 0。 

 public static final int ChangLiang = 666;

之所以 static final 会直接被复制,而 static 变量会被赋予java语言类型的默认值。其实我们稍微思考一下就能想明白了。 

两个语句的区别是一个有 final 关键字修饰,另外一个没有。而 final 关键字在 Java 中代表不可改变的意思,意思就是说 ChangLiang的值一旦赋值就不会在改变了。既然一旦赋值就不会再改变,那么就必须一开始就给其赋予用户想要的值,因此被 final 修饰的类变量在准备阶段就会被赋予想要的值。而没有被 final 修饰的类变量,其可能在初始化阶段或者运行阶段发生变化,所以就没有必要在准备阶段对它赋予用户想要的值。 

如果还不是很清晰理解final和static关键字的话建议参阅下面博主整理好的文章,希望对你有所帮助! 

 java中的Static、final、Static final各种用法 

2、4.解析 

当通过准备阶段之后,进入解析阶段。解析阶段是虚拟机将常量池内的符号引用替换为直接引用的过程,解析动作主要针对类或接口、字段、类方法、接口方法、方法类型、方法句柄和调用点限定符7类符号引用进行。符号引用就是一组符号来描述目标,可以是任何字面量。 

直接引用就是直接指向目标的指针、相对偏移量或一个间接定位到目标的句柄。 

其实这个阶段对于我们来说也是几乎透明的,了解一下就好。 

2、5.初始化【重点】 

到了初始化阶段,用户定义的 Java 程序代码才真正开始执行。 

Java程序对类的使用方式可分为两种:主动使用与被动使用。一般来说只有当对类的首次主动使用的时候才会导致类的初始化,那啥是主动使用呢?类的主动使用包括以下六种【重点】:

 1、 创建类的实例,也就是new的方式 
 2、 访问某个类或接口的静态变量,或者对该静态变量赋值(凡是被final修饰不不不其实更准确的说是在编译器把结果放入常量池的静态字段除外) 
 3、 调用类的静态方法 
 4、 反射(如 Class.forName(“com.gx.yichun”)) 
 5、 初始化某个类的子类,则其父类也会被初始化 
 6、 Java虚拟机启动时被标明为启动类的类( JavaTest ),还有就是Main方法的类会首先被初始化 
最后注意一点对于静态字段,只有直接定义这个字段的类才会被初始化(执行静态代码块),这句话在继承、多态中最为明显!为了方便理解下文会陆续通过例子讲解 

2、6.使用 

当 JVM 完成初始化阶段之后,JVM 便开始从入口方法开始执行用户的程序代码。这个使用阶段也只是了解一下就可以了。 

2、7.卸载 

当用户程序代码执行完毕后,JVM 便开始销毁创建的 Class 对象,最后负责运行的 JVM 也退出内存。这个卸载阶段也只是了解一下就可以了。 

2、8.结束生命周期 

在如下几种情况下,Java虚拟机将结束生命周期 

1、 执行了 System.exit()方法 
2、 程序正常执行结束 
3、 程序在执行过程中遇到了异常或错误而异常终止 
4、 由于操作系统出现错误而导致Java虚拟机进程终止 

解开开篇的面试题 

package com.jvm.classloader;

class Father2{
    public static String strFather="HelloJVM_Father";

    static{
        System.out.println("Father静态代码块");
    }
}

class Son2 extends Father2{
    public static String strSon="HelloJVM_Son";

    static{
        System.out.println("Son静态代码块");
    }
}

public class InitativeUseTest2 {
    public static void main(String[] args) {

       System.out.println(Son2.strSon);
    }
}

运行结果:
        Father静态代码块
        Son静态代码块
        HelloJVM_Son

再回头看这个题,这也太简单了吧,由于Son2.strSon是调用了Son类自己的静态方法属于主动使用,所以会初始化Son类,又由于继承关系,类继承原则是初始化一个子类,会先去初始化其父类,所以会先去初始化父类! 

再看开篇的第二个题 

 package com.jvm.classloader;

class YeYe{
    static {
        System.out.println("YeYe静态代码块");
    }
}

class Father extends YeYe{
    public static String strFather="HelloJVM_Father";

    static{
        System.out.println("Father静态代码块");
    }
}

class Son extends Father{
    public static String strSon="HelloJVM_Son";

    static{
        System.out.println("Son静态代码块");
    }
}

public class InitiativeUse {
    public static void main(String[] args) {
        System.out.println(Son.strFather); 
    }
}

运行结果:
	YeYe静态代码块
    Father静态代码块
    HelloJVM_Father

这个题就稍微要注意一下,不过要是你看懂这篇文章,这个题也很简单。这个题要注意什么呢?要注意子类Son类没有被初始化,也就是Son的静态代码块没有执行!发现了咩?那我们来分析分析… 

首先看到Son.strFather,你会发现是子类Son访问父类Father的静态变量strFather,这个时候就千万要记住我在归纳主动使用概念时特别提到过的一个注意点了:对于静态字段,只有直接定义这个字段的类才会被初始化(执行静态代码块),这句话在继承、多态中最为明显! 

嗯哼,对吧,Son.strFather中的静态字段是属于父类Father的对吧,也就是说直接定义这个字段的类是父类Father,所以在执行 System.out.println(Son.strFather); 这句代码的时候会去初始化Father类而不是子类Son!是不是一下子明白了?如果明白了就支持一下小瓜哥点个赞呗,谢谢~

再看开篇的第三个题 

 package com.jvm.classloader;

class YeYe{
    static {
        System.out.println("YeYe静态代码块");
    }
}

class Father extends YeYe{
    public final static String strFather="HelloJVM_Father";

    static{
        System.out.println("Father静态代码块");
    }
}

class Son extends Father{
    public static String strSon="HelloJVM_Son";

    static{
        System.out.println("Son静态代码块");
    }
}

public class InitiativeUse {
    public static void main(String[] args) {
        System.out.println(Son.strFather);
    }
}

运行结果:HelloJVM_Father

  • 这个题唯一的特点就在于final static !是的Son.strFather所对应的变量便是final static修饰的,依旧是在本篇文章中归纳的类的主动使用范畴第二点当中:访问某个类或接口的静态变量,或者对该静态变量赋值(凡是被final修饰不不不其实更准确的说是在编译器把结果放入常量池的静态字段除外) 
  • 所以,这个题并不会初始化任何类,当然除了Main方法所在的类!于是仅仅执行了System.out.println(Son.strFather);所以仅仅打印了Son.strFather的字段结果HelloJVM_Father,嗯哼,是不是又突然明白了?如果明白了就再支持一下小瓜哥点个赞呗,谢谢~ 
  • 实际上上面的题目并不能完全说明本篇文章中归纳的类的主动使用范畴第二点!这话怎么说呢?怎么理解呢?再来一个程序各位就更加明了了 

 package com.jvm.classloader;

import sun.applet.Main;

import java.util.Random;
import java.util.UUID;

class Test{
    static {
        System.out.println("static 静态代码块");
    }

//    public static final String str= UUID.randomUUID().toString();
    public static final double str=Math.random();  //编译期不确定
}


public class FinalUUidTest {
    public static void main(String[] args) {
        System.out.println(Test.str);
    }
}

请试想一下结果,会不会执行静态代码块里的内容呢? 

重点来了 

重点来了 

重点来了 

运行结果 

static 静态代码块
0.7338688977344875

  • 上面这个程序完全说明本篇文章中归纳的类的主动使用范畴第二点当中的这句话:凡是被final修饰不不不其实更准确的说是在编译器把结果放入常量池的静态字段除外! 
  • 分析:其实final不是重点,重点是编译器把结果放入常量池!当一个常量的值并非编译期可以确定的,那么这个值就不会被放到调用类的常量池中,这时在程序运行时,会导致主动使用这个常量所在的类,所以这个类会被初始化 
  • 到这里,能理解完上面三个题已经很不错了,但是要想更加好好的学习java,博主不得不给各位再来一顿烧脑盛宴,野心不大,只是单纯的想巅覆各位对java代码的认知,当然还望大佬轻拍哈哈哈,直接上代码: 

 package com.jvm.classloader;

public class ClassAndObjectLnitialize {

        public static void main(String[] args) {
            System.out.println("输出的打印语句");
        }

      public ClassAndObjectLnitialize(){

            System.out.println("构造方法");
            System.out.println("我是熊孩子我的智商=" + ZhiShang +",情商=" + QingShang);
        }

        {
            System.out.println("普通代码块");
        }

        int ZhiShang = 250;
        static int QingShang = 666;
        
        static
        {
            System.out.println("静态代码块");
        }     

}

建议这个题不要花太多时间思考,否则看了结果你会发现自己想太多了,导致最后可能你看到结果想砸电脑

 隔离运行结果专业跑龙套… 

 隔离运行结果专业跑龙套… 

 隔离运行结果专业跑龙套… 

 隔离运行结果专业跑龙套… 

运行结果
		静态代码块
		输出的打印语句

怎么样,是不是没有你想的那么复杂呢? 

下面我们来简单分析一下,首先根据上面说到的触发初始化的(主动使用)的第六点:Java虚拟机启动时被标明为启动类的类( JavaTest ),还有就是Main方法的类会首先被初始化 

嗯哼?小白童鞋就有疑问了:不是说好有Main方法的类会被初始化的么?那怎么好多东西都没有执行捏? 

那么类的初始化顺序到底是怎么样的呢?在我们代码中,我们只知道有一个构造方法,但实际上Java代码编译成字节码之后,最开始是没有构造方法的概念的,只有类初始化方法 和 对象初始化方法 。 

这个时候我们就不得不深入理解了!那么这两个方法是怎么来的呢?

类初始化方法:编译器会按照其出现顺序,收集:类变量(static变量)的赋值语句、静态代码块,最终组成类初始化方法。类初始化方法一般在类初始化的时候执行。 

所以,上面的这个例子,类初始化方法就会执行下面这段代码了: 

 static int QingShang = 666;  //类变量(static变量)的赋值语句

  static   //静态代码块
   {
       System.out.println("静态代码块");
   }

而不会执行普通赋值语句以及普通代码块了 

对象初始化方法:编译器会按照其出现顺序,收集:成员变量的赋值语句、普通代码块,最后收集构造函数的代码,最终组成对象初始化方法,值得特别注意的是,如果没有监测或者收集到构造函数的代码,则将不会执行对象初始化方法。对象初始化方法一般在实例化类对象的时候执行。 

以上面这个例子,其对象初始化方法就是下面这段代码了: 

 {                        
       System.out.println("普通代码块");    //普通代码块
    }
 
    int ZhiShang = 250;   //成员变量的赋值语句
    
    System.out.println("构造方法");  //最后收集构造函数的代码
    System.out.println("我是熊孩子我的智商=" + ZhiShang +",情商=" + QingShang);

明白了类初始化方法 和 对象初始化方法 之后,我们再来看这个上面例子!是的!正如上面提到的:如果没有监测或者收集到构造函数的代码,则将不会执行对象初始化方法。上面的这个例子确实没有执行对象初始化方法。忘了吗?我们根本就没有对类ClassAndObjectLnitialize 进行实例化!只是单纯的写了一个输出语句。 

如果我们给其实例化,验证一下,代码如下:

 package com.jvm.classloader;

public class ClassAndObjectLnitialize {

        public static void main(String[] args) {
            new ClassAndObjectLnitialize();
            System.out.println("输出的打印语句");
        }

      public ClassAndObjectLnitialize(){

            System.out.println("构造方法");
             System.out.println("我是熊孩子我的智商=" + ZhiShang +",情商=" + QingShang);
        }

        {
            System.out.println("普通代码块");
        }

        int ZhiShang = 250;
        static int QingShang = 666;
        
        static
        {
            System.out.println("静态代码块");
        }      
}

运行结果:
		静态代码块
		普通代码块
		构造方法
		我是熊孩子我的智商=250,情商=666
		输出的打印语句

总结 

类的加载、连接与初始化: 

 1、加载:查找并加载类的二进制数据到java虚拟机中

 

2、 连接:

验证: 确保被加载的类的正确性
准备:为类的静态变量分配内存,并将其初始化为默认值,但是到达初始化之前类变量都没有初始化为真正的初始值(如果是被 final 修饰的类变量,则直接会被初始成用户想要的值。)
解析:把类中的符号引用转换为直接引用,就是在类型的常量池中寻找类、接口、字段和方法的符号引用,把这些符号引用替换成直接引用的过程

3、 初始化:为类的静态变量赋予正确的初始值

 

类从磁盘上加载到内存中要经历五个阶段:加载、连接、初始化、使用、卸载 

Java程序对类的使用方式可分为两种 

 (1)主动使用
(2)被动使用 

所有的Java虚拟机实现必须在每个类或接口被Java程序“首次主动使用”时才能初始化他们 

主动使用 

  • (1)创建类的实例 
  • (2)访问某个类或接口的静态变量 getstatic(助记符),或者对该静态变量赋值 putstatic (3)调用类的静态方法 invokestatic 
  • (4)反射(Class.forName(“com.test.Test”)) 
  • (5)初始化一个类的子类 
  • (6)Java虚拟机启动时被标明启动类的类以及包含Main方法的类 
  • (7)JDK1.7开始提供的动态语言支持(了解) 

被动使用 

除了上面七种情况外,其他使用java类的方式都被看做是对类的被动使用,都不会导致类的初始化 

特别注意 

  • 初始化入口方法。当进入类加载的初始化阶段后,JVM 会寻找整个 main 方法入口,从而初始化 main 方法所在的整个类。当需要对一个类进行初始化时,会首先初始化类构造器(),之后初始化对象构造器()。 
  • 初始化类构造器:JVM 会按顺序收集类变量的赋值语句、静态代码块,最终组成类构造器由 JVM 执行。 
  • 初始化对象构造器:JVM 会按照收集成员变量的赋值语句、普通代码块,最后收集构造方法,将它们组成对象构造器,最终由 JVM 执行。值得特别注意的是,如果没有监测或者收集到构造函数的代码,则将不会执行对象初始化方法。对象初始化方法一般在实例化类对象的时候执行。 
  • 如果在初始化 main 方法所在类的时候遇到了其他类的初始化,那么就先加载对应的类,加载完成之后返回。如此反复循环,最终返回 main 方法所在类。 

当然这篇文章还没讲到类加载器以及双亲委派模型等,之后的时间再总结一下! 

如果本文对你有一点点帮助,那么请点个赞,谢谢~ 

最后

喜欢的可以加我公众号“java小瓜哥的分享平台”免费分享各种技术文,共同进步!