《JVM由浅入深学习【一】 》JVM由简入深学习提升(类加载过程+父子类加载过程+类加载器+双亲委派机制)

简介: 《JVM由浅入深学习【一】 》JVM由简入深学习提升(类加载过程+父子类加载过程+类加载器+双亲委派机制)

JVM的类加载

1. java运行时是什么时候被加载的?

我们现在用的一般是HotSpot虚拟机,它是按需加载的,也就是说,在需要用到这个类的时候再去加载。

2. JVM类加载过程大致阶段

加载–》验证–》准备–》解析–》初始化–》使用–》卸载

其中验证–》准备–》解析 总结为链接

(1)加载:将我们classpath下的class文件的二进制字节流读出来,此过程我们可以自定义类加载器来实现类的加载。

(2)验证:验证Class文件的字节流中包含的信息是否符合《Java虚拟机规范》的全部约束要求,保证虚拟机的安全。

(3)准备:类变量赋初始值,int为0,long为0L,boolean为false,引用类型为null,常量赋正式值

(4)解析:把符号引用变为直接引用。

(5)初始化:当我们new一个类的对象,访问一个类的静态属性,修改一个类的静态属性,调用一个静态类的方法,用反射api调用一个类,初始化当前类,其父类也会初始化…这些都会触发类的初始化

(6)使用:使用这个类

(7)卸载:(需要同时满足以下三个条件)

- 该类的所有实例都被GC,也就是JVM不存在任何该类的实例

- 加载该类的classLoader已经被GC了

- 该类的java.lang.class对像没有在任何地方被引用,并且不能在任何地方通过反射的方法访问该类的方法。

3. 父类与子类初始化各个类型顺序

父类和子类都有(静态变量,变量,静态初始化块,初始块,构造函数)

public class People {
    // 静态变量
    private static String p_StaticField = "父类----静态变量";
    // 变量
    private String p_Field = "父类----变量";
    protected int i = 0;
    protected int j = 0;
    // 静态初始化块
    static {
        System.out.println(p_StaticField);
        System.out.println("父类----静态初始块");
    }
    // 初始快
    {
        System.out.println(p_Field);
        System.out.println("父类----初始块");
    }
    // 构造函数
    public People() {
        System.out.println("父类----构造函数");
        System.out.println("i=" + i + "j=" + j
        );
    }
}
public class Student extends People {
    // 静态变量
    private static String p_StaticField = "子类----静态变量";
    // 变量
    private String p_Field = "子类----变量";
    protected int i = 0;
    protected int j = 0;
    // 静态初始化块
    static {
        System.out.println(p_StaticField);
        System.out.println("子类----静态初始块");
    }
    // 初始快
    {
        System.out.println(p_Field);
        System.out.println("子类----初始块");
    }
    // 构造函数
    public Student() {
        System.out.println("子类----构造函数");
        System.out.println("i=" + i + "j=" + j
        );
    }
    public static void main(String[] args) {
//        new Student();
    }
}

(1)不new子类,运行main。加载顺序如下(先加载父类静态变量和静态初始块,再加载子类静态变量和静态初始块)

父类----静态变量
父类----静态初始块
子类----静态变量
子类----静态初始块

(2)new 子类(创建对象),顺序如下(在之前加载基础上,先初始化父类,再是子类)

父类----静态变量
父类----静态初始块
子类----静态变量
子类----静态初始块
父类----变量
父类----初始块
父类----构造函数i=0j=0
子类----变量
子类----初始块
子类----构造函数i=0j=0

(3)在子类main中创建父类对象(new 父类,在之前基础上,再加载父类其他)

父类----静态变量
父类----静态初始块
子类----静态变量
子类----静态初始块
父类----变量
父类----初始块
父类----构造函数i=0j=0

(4)在父类main中创建父类对象(只加载父类,不加载子类)

父类----静态变量
父类----静态初始块
父类----变量
父类----初始块
父类----构造函数i=0j=0

4. 什么是类加载器?

在类 加载 阶段,通过 一个类的全限定名 来获取 描述该类的二进制字节流的这个动作 的 代码 被称为 类加载器,该动作可以自定义。

分两大类:(四层结构 启动类加载器–》拓展类加载器–》应用程序加载器–》自定义加载器)

  • 启动类加载器(Bootstrap ClassLoader)由C++实现,是虚拟机自身的一部分
  • 其他类加载器:主要是由java实现的,独立于虚拟机之外的,都继承自抽象类(java.lang.ClassLoader)


    下面三个例子(类.class.getClassLoader()可以获取类加载器``)
public class People {
    public static void main(String[] args) {
        System.out.println(BufferedReader.class.getClassLoader());
        System.out.println(DNSNameService.class.getClassLoader());
        System.out.println(People.class.getClassLoader());
    }
}

结果:

null           // 为空说明为启动类加载器加载(因为是在java虚拟机里面,所以为空)
sun.misc.Launcher$ExtClassLoader@49476842 
sun.misc.Launcher$AppClassLoader@18b4aac2
  1. 类加载器的三层结构是继承关系么?
    不是,首先看ExtClassLoader和AppClassLoader,都属于Launcher静态内部类,而BootStrapClassLoader是C++书写的,java更不可能继承。以及我们自定义的类加载器,其实也是继承ClassLoader的。

6. 双亲委派机制

(1)这是一个加载流程(先踢皮球,后各司其职),先自底向上委派到BootStrapClassLoader,再自顶向下找,属于谁的任务,给谁加载。都找不到就报ClassNotFound异常

(2)为什么设计双亲委派机制

  • 确保安全,避免java核心类库被修改
  • 避免重复加载
  • 保证类的唯一性
    (3)是否可以打破双亲委派机制
    可以,想要打破,可以自定义类加载器,重写loadClass方法,使其不进行双亲委派即可
相关文章
|
1月前
|
缓存 算法 Java
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
这篇文章详细介绍了Java虚拟机(JVM)中的垃圾回收机制,包括垃圾的定义、垃圾回收算法、堆内存的逻辑分区、对象的内存分配和回收过程,以及不同垃圾回收器的工作原理和参数设置。
60 4
JVM知识体系学习六:JVM垃圾是什么、GC常用垃圾清除算法、堆内存逻辑分区、栈上分配、对象何时进入老年代、有关老年代新生代的两个问题、常见的垃圾回收器、CMS
|
1月前
|
存储 SQL 小程序
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
这篇文章详细介绍了Java虚拟机(JVM)的运行时数据区域和JVM指令集,包括程序计数器、虚拟机栈、本地方法栈、直接内存、方法区和堆,以及栈帧的组成部分和执行流程。
31 2
JVM知识体系学习五:Java Runtime Data Area and JVM Instruction (java运行时数据区域和java指令(大约200多条,这里就将一些简单的指令和学习))
|
1月前
|
Java 应用服务中间件 程序员
JVM知识体系学习八:OOM的案例(承接上篇博文,可以作为面试中的案例)
这篇文章通过多个案例深入探讨了Java虚拟机(JVM)中的内存溢出问题,涵盖了堆内存、方法区、直接内存和栈内存溢出的原因、诊断方法和解决方案,并讨论了不同JDK版本垃圾回收器的变化。
29 4
|
1月前
|
Arthas 监控 Java
JVM知识体系学习七:了解JVM常用命令行参数、GC日志详解、调优三大方面(JVM规划和预调优、优化JVM环境、JVM运行出现的各种问题)、Arthas
这篇文章全面介绍了JVM的命令行参数、GC日志分析以及性能调优的各个方面,包括监控工具使用和实际案例分析。
43 3
|
1月前
|
存储 Java
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
这篇文章详细地介绍了Java对象的创建过程、内存布局、对象头的MarkWord、对象的定位方式以及对象的分配策略,并深入探讨了happens-before原则以确保多线程环境下的正确同步。
53 0
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
|
1月前
|
存储 安全 Java
jvm 锁的 膨胀过程?锁内存怎么变化的
【10月更文挑战第3天】在Java虚拟机(JVM)中,`synchronized`关键字用于实现同步,确保多个线程在访问共享资源时的一致性和线程安全。JVM对`synchronized`进行了优化,以适应不同的竞争场景,这种优化主要体现在锁的膨胀过程,即从偏向锁到轻量级锁,再到重量级锁的转变。下面我们将详细介绍这一过程以及锁在内存中的变化。
37 4
|
6天前
|
Arthas 监控 Java
JVM进阶调优系列(9)大厂面试官:内存溢出几种?能否现场演示一下?| 面试就那点事
本文介绍了JVM内存溢出(OOM)的四种类型:堆内存、栈内存、元数据区和直接内存溢出。每种类型通过示例代码演示了如何触发OOM,并分析了其原因。文章还提供了如何使用JVM命令工具(如jmap、jhat、GCeasy、Arthas等)分析和定位内存溢出问题的方法。最后,强调了合理设置JVM参数和及时回收内存的重要性。
|
4天前
|
Java Linux Windows
JVM内存
首先JVM内存限制于实际的最大物理内存,假设物理内存无限大的话,JVM内存的最大值跟操作系统有很大的关系。简单的说就32位处理器虽然可控内存空间有4GB,但是具体的操作系统会给一个限制,这个限制一般是2GB-3GB(一般来说Windows系统下为1.5G-2G,Linux系统下为2G-3G),而64bit以上的处理器就不会有限制。
7 1
|
1月前
|
存储 缓存 算法
JVM核心知识点整理(内存模型),收藏再看!
JVM核心知识点整理(内存模型),收藏再看!
JVM核心知识点整理(内存模型),收藏再看!
|
23天前
|
存储 算法 Java
聊聊jvm的内存结构, 以及各种结构的作用
【10月更文挑战第27天】JVM(Java虚拟机)的内存结构主要包括程序计数器、Java虚拟机栈、本地方法栈、Java堆、方法区和运行时常量池。各部分协同工作,为Java程序提供高效稳定的内存管理和运行环境,确保程序的正常执行、数据存储和资源利用。
45 10