小米科技|看看96后小伙是怎么拿捏腾讯面试官拿到offer的( 四 )


  • 扩展类加载器(Extension or Ext Class-Loader) , 加载 jre/lib/ext 包下面的 jar 文件 。
  • 应用类加载器(Application or App Clas-Loader) , 根据程序的类路径(classpath)来加载 Java 类 。
  • 一般来说 , Java 程序员并不需要直接同类加载器进行交互 。 JVM 默认的行为就已经足够满足大多数情况的需求了 。 不过 , 如果遇到了需要和类加载器进行交互的情况 , 而对类加载器的机制又不是很了解的话 , 就不得不花大量的时间去调试
    ClassNotFoundException 和 NoClassDefFoundError 等异常 。
    对于任意一个类 , 都需要由它的类加载器和这个类本身一同确定其在 JVM 中的唯一性 。 也就是说 , 如果两个类的加载器不同 , 即使两个类来源于同一个字节码文件 , 那这两个类就必定不相等(比如两个类的 Class 对象不 equals) 。
    是不是有点晕 , 来来来 , 通过一段简单的代码了解下 。

    每个Java 类都维护着一个指向定义它的类加载器的引用 , 通过 类名.class.getClassLoader() 可以获取到此引用;然后通过 loader.getParent() 可以获取类加载器的上层类加载器 。
    上面这段代码的输出结果如下:

    第一行输出为 Test 的类加载器 , 即应用类加载器 , 它是 sun.misc.Launcher$AppClassLoader 类的实例;第二行输出为扩展类加载器 , 是 sun.misc.Launcher$ExtClassLoader 类的实例 。 那启动类加载器呢?
    按理说 , 扩展类加载器的上层类加载器是启动类加载器 , 但在我这个版本的 JDK 中 ,扩展类加载器的 getParent() 返回 null 。 所以没有输出 。
    2)运行时数据区
    运行时数据区又包含以下内容:

    • PC寄存器(PC Register) , 也叫程序计数器(Program Counter Register) , 是一块较小的内存空间 , 它的作用可以看做是当前线程所执行的字节码的信号指示器 。
    • JVM 栈(Java Virtual Machine Stack) , 与 PC 寄存器一样 , JVM 栈也是线程私有的 。 每一个 JVM 线程都有自己的 JVM 栈 , 这个栈与线程同时创建 , 它的生命周期与线程相同 。
    • 本地方法栈(Native Method Stack) , JVM 可能会使用到传统的栈来支持 Native 方法(使用 Java 语言以外的其它语言[C语言]编写的方法)的执行 , 这个栈就是本地方法栈 。
    • 堆(Heap) , 在 JVM 中 , 堆是可供各条线程共享的运行时内存区域 , 也是供所有类实例和数据对象分配内存的区域 。
    • 方法区(Method area) , 在 JVM 中 , 被加载类型的信息都保存在方法区中 。 包括类型信息(Type Information)和方法列表(Method Tables) 。 方法区是所有线程共享的 , 所以访问方法区信息的方法必须是线程安全的 。
    • 运行时常量池(Runtime Constant Pool) , 运行时常量池是每一个类或接口的常量池在运行时的表现形式 , 它包括了编译器可知的数值字面量 , 以及运行期解析后才能获得的方法或字段的引用 。 简而言之 , 当一个方法或者变量被引用时 , JVM 通过运行时常量区来查找方法或者变量在内存里的实际地址 。
    3)执行引擎
    执行引擎包含了: