《深入解析Java虚拟机:从JVM体系结构到垃圾回收算法》

文章目录

    • JVM体系结构
      • JVM的组成
    • 类加载器Class Loader
      • 类加载器的作用
        • 双亲委派机制
          • JVM自带三个类加载器
            • Bootstrap ClassLoader-根加载器
            • ExtClassLoader-扩展加载器
            • AppClassLoader-应用类加载器
    • Java历史-沙箱安全机制
      • 沙箱概念
      • 沙箱的作用
      • 本地代码和远程代码
      • 沙箱安全机制模型
        • JDK1 .0安全模型
        • JDK1 .1安全模型
        • JDK1 .2安全模型
        • 目前最新的安全模型
      • 沙箱安全机制的基本组件
        • 字节码校验器(bytecode verifier)
        • 类装载器(class loader)
        • 存取控制器(access controller)
        • 安全管理器(security manager)
        • 安全软件包(security package) :
    • Native与方法区
      • Native
      • 方法区
    • 深入理解栈
      • 关于main方法的调用
      • 栈运行的原理
        • 栈帧
    • HotSpot和堆
    • 新生区、老年区、永久区
      • 新生区(伊甸园区+幸存者区*2)
      • 老年区
      • 永久区
    • 堆内存调优
      • 报OOM怎么办?
        • 使用Jprofiler工具分析OOM原因
    • imgGC:垃圾回收
      • GC算法
        • 引用计数法
        • 复制算法
        • 标记清除算法
          • 标记清除
          • 标记清除压缩
      • 总结

来看一下常见的JVM面试题:

  • 请你谈谈对JVM的理解?java8虚拟机和之前的变化更新?
  • 什么是OOM,什么是栈溢出StackOverFlowError?怎么分析?
  • JVM的常用调优参数有哪些?
  • 内存快照如何抓取,怎么分析Dump文件?
  • 谈谈你对JVM中类加载器的认识?

如果回答不了上面的题目,那就开始接下来的学习JVM之路把~

JVM体系结构

Java程序是跑在我们的JVM虚拟机上的,而虚拟机又在操作系统之上。

下面的图要自己动手画一遍!

img

需要知道,操作系统是在硬件之上的。

img

栈这个区域是不会有垃圾的:

img

垃圾都在堆区,而方法区又是特殊堆:

img

jvm调优:99%都是在方法区和堆,大部分时间调堆。

img

JVM的组成

通过上图,我们已知JVM由三部分组成:

  • 类加载子系统
  • 运行时数据区
  • 执行引擎

类加载器Class Loader

类加载器的作用

类加载器的作用是把类(class)装在进内存里。

package com.linghu;

/**
 * @author linghu
 * @date 2024/1/19 16:57
 */
public class Car {
    public static void main(String[] args) {
        //类是模板,对象是具体的
        Car car1 = new Car();
        Car car2 = new Car();
        Car car3 = new Car();

        System.out.println("======对象=======");
        System.out.println(car1.hashCode());
        System.out.println(car2.hashCode());
        System.out.println(car3.hashCode());

        //这就是类的模板
        Class<? extends Car> aClass1 = car1.getClass();
        Class<? extends Car> aClass2 = car1.getClass();
        Class<? extends Car> aClass3 = car1.getClass();
        System.out.println("======类模板=======");
        System.out.println(aClass1.hashCode());
        System.out.println(aClass2.hashCode());
        System.out.println(aClass3.hashCode());
    }
}

分析如上代码和下图,可知:

类加载器Class Loader将类Car加载初始化成类模板。 也就是上面的aClass1aClass2aClass3 就是类模板,这三个都是同一个类模板

我们输出类模板的hash:

img

类模板的作用就是,类模板可以为我们的类实例化对象,也就是创建对象!我们上面的代码已经创建了三个对象:car1car2car3

我们输出三个对象的hash:

img

同样,我们也可以通过 类模板调用getClassLoader()得到类加载器,从而获取类的名字,这就是Java反射!

img

双亲委派机制

首先需要知道的是ClassLoader的作用就是将class文件加载到jvm虚拟机中去,JVM就可以正确运行了。但是,jvm启动的时候,并不会一次性加载所有的class文件,而是根据需要去动态加载。

  1. 类加载器在加载类的时候,优先委托给上级类加载器,其委托方向如为 AppClassLoader -> ExtClassLoader -> BootStrap

JVM自带的有三个类加载器:

JVM自带三个类加载器
Bootstrap ClassLoader-根加载器

最顶层的加载类。加载目录jre/lib/rt.jar里所有的class。

ExtClassLoader-扩展加载器

扩展的类加载器,加载目录%JRE_HOME%\lib\ext目录下的jar包和class文件。

AppClassLoader-应用类加载器

加载当前应用的classpath的所有类。 也就是我们自己定义的class。

  • 初始化顺序: Bootstrap ClassLoader->ExtClassLoader(加载路径:java.ext.dirs)->AppClassLoader(加载路径:java.class.path) 。
package com.linghu;

/**
 * @author linghu
 * @date 2024/1/19 16:57
 */
public class Car {
    public static void main(String[] args) {
        //类是模板,对象是具体的
        Car car1 = new Car();
        Car car2 = new Car();
        Car car3 = new Car();

//        System.out.println("======对象=======");
//        System.out.println(car1.hashCode());
//        System.out.println(car2.hashCode());
//        System.out.println(car3.hashCode());

        //这就是类的模板
        Class<? extends Car> aClass1 = car1.getClass();
//        Class<? extends Car> aClass2 = car1.getClass();
//        Class<? extends Car> aClass3 = car1.getClass();
        ClassLoader classLoader = aClass1.getClassLoader();
        System.out.println(classLoader);//AppClassLoader
        System.out.println(classLoader.getParent());//ExtClassLoader-->\jre\lib\ext
        System.out.println(classLoader.getParent().getParent());//null,1、不存在

    }
}

img

img

总结双亲委派保证类加载器,自下而上的委派,又自上而下的加载,保证每一个类在各个类加载器中都是同一个类。一个非常明显的目的就是保证java官方的类库<JAVA_HOME>\lib和扩展类库<JAVA_HOME>\lib\ext的加载安全性,不会被开发者覆盖。

Java历史-沙箱安全机制

沙箱概念

Java安全模型的核心就是Java沙箱(sandbox)。沙箱机制就是将Java代码限定只能在虚JVM虚拟机中特定的运行范围,并且严格限制代码对本地系统资源访问,通过这样的方式来保证对Java代码的有效隔离,防止对本地操作系统造成破坏。

其实Windows也有这个沙箱的概念,像个内置虚拟机。

沙箱的作用

主要限制系统资源(CPU、内存、文件系统、网络)的访问。

不同级别的沙箱对系统资源访问的限制也有差异。

本地代码和远程代码

Java的执行程序分为:本地代码和远程代码。,

  • 本地代码:默认视为可信任的,可以访问一切本地资源。

  • 远程代码:被看作是不受信的。对于授信的本地代码,对于非授信的远程代码在早期的Java实现中,安全依赖于沙箱(Sandbox)机制。

沙箱安全机制模型

JDK1 .0安全模型

JDK1 .0安全模型本地代码可以访问系统资源,远程代码无法访问系统资源,比如用户希望远程代码访问本地系统的文件时候,就无法实现。

img

JDK1 .1安全模型

JDK1 .1 安全模型版本中,针对安全机制做了改进,增加了受信任安全策略,允许用户指定代码对本地资源的访问权限

img

JDK1 .2安全模型

JDK1 .2安全模型改进了安全机制,增加了代码签名。不论本地代码或是远程代码,统一按照用户的安全策略设定,由类加载器加载到虚拟机中权限不同的运行空间,从而来实现差异化的代码执行权限控制。

img

目前最新的安全模型

目前最新的安全模型引入了域 (Domain) 的概念。JVM虚拟机会把所有代码加载到不同的系统域和应用域,系统域部分专门负责与关键资源系统进行交互,而每个应用域部分则通过系统域的部分代理来对各种需要的资源进行精细划分然后可以进行访问。JVM虚拟机中不同的受保护域 (Protected Domain)对应不一样的权限 (Permission)。存在于不同域中的类文件就拥有了它所包含应用域所有可访问资源之和。

img

沙箱安全机制的基本组件

字节码校验器(bytecode verifier)

确保lava类文件遵循lava语言规范。这样可以帮助Java程序实现内存保护。但并不是所有的类文件都会经过字节码校验,比如核心类。

类装载器(class loader)
  • 防止恶意代码去干涉善意的代码,比如:双亲委派机制
  • 守护了被信任的类库边界;
  • 将代码归入保护域,确定了代码的权限范围可以进行哪些资源操作
存取控制器(access controller)

存取控制器可以控制核心API对操作系统的存取权限,用户可以设定控制策略。

安全管理器(security manager)

安全管理器主要是核心API和操作系统之间的主要接口。比如实现权限控制,比存取控制器优先级高。

安全软件包(security package) :

java.security下的类和扩展包下的类,允许用户为应用增加所需要安全特性:安全提供者、消息摘要、数字签名keytools、加密、鉴别。

Native与方法区

Native

package com.linghu;

/**
 * @author linghu
 * @date 2024/1/23 9:53
 */
public class Demo {
    public static void main(String[] args) {
        new Thread(()->{

        },"myThread").start();
    }

    private native void start0();
}

通过以上代码,可以看到有一个 start0方法。该方法用 native来进行修饰。有以下含义:

  • 凡事有 native修饰的,说明Java作用范围达不到了。回去调用C语言的库了~
  • 会进入本地方法栈
  • 调用本地方法接口JNI
private native void start0();

JNI作用:扩展Java的使用,融合不同的编程语言为Java所用!

img

Java诞生的时候,正是C和C++大行其道的时候,为了能够在市场上存活,Java在内存中单独开了一个标记区本地方法栈 Native Method Stack,登记Native方法,在最终执行的时候,加载本地方法库中的方法通过JNI。

img

Java程序驱动打印机、管理系统等等都需要用到这个Native。在企业级应用中较为少见!

方法区

Method Area 方法区。
方法区是被所有线程共享,所有字段和方法字节码,以及一些特殊方法,如构造函数,接口代码也在此定义简单说,所有定义的方法的信息都保存在该区域,此区域属于共享区间;

静态变量、常量、类信息 (构造方法、接口定义)、运行时的常量池存在方法区中,但是 实例变量存在堆内存中,和方法区无关
static, final,Class,常量池

深入理解栈

栈:又叫栈内存。主管程序的运行和生命周期、线程同步。

线程结束,栈内存也就释放了。对于栈来说,不存在垃圾回收的说法、一旦线程结束,栈就Over!

栈内存中有:8大基本类型+对象引用+实例的方法。

关于main方法的调用

为什么main先执行,最后结束?

我们调用一个main方法,在main里调用test(),在test()调用a()方法,会出现内存溢出的问题,这个问题用栈表示:

img

test和a方法循环调用对方,最后会出现栈溢出的问题,也就是内存的问题,因为这个调用是没有限制的,无限循环!

img

栈运行的原理

栈帧

一个方法对应一个栈帧!其实这个可以理解成一个FCB!下面是两个栈帧!

栈帧(Stack Frame)是用于支持虚拟机进行方法调用和方法执行的数据结构。栈帧存储了方法的局部变量表、操作数栈、动态连接和方法返回地址等信息。每一个方法从调用至执行完成的过程,都对应着一个栈帧在虚拟机栈里从入栈到出栈的过程。

img

img

HotSpot和堆

我们通过 java -version命令查看:

img

我们用的虚拟机一般都是 HotSpot

Heap,一个JVM只有一个堆内存。堆内存的大小是可以调节的。

类加载器读取类文件后,一般会把什么东西放到堆中?

类、方法、常量、变量~保存所有引用类型的真实对象。

堆内存细分三个区域:

  • 新生区(伊甸园区)
  • 养老区
  • 永久区

堆也被称作GC堆

img

GC垃圾回收,主要是在伊甸园区和养老区~假设内存满了,报错OOM,堆内存不够!

img

JDK8以后,永久存储区改了个名字,叫元空间

新生区、老年区、永久区

其实通过上面的图就会发现:

新生区(伊甸园区+幸存者区*2)

  • 类诞生和死亡的地方
  • 伊甸园区:所有对象都是伊甸园区new出来的!
  • 幸存者区(0、1),轻GC定期清理伊甸园区,活下来的放入幸存者区,幸存者区满了以后重GC清理伊甸园区+幸存者区。活下来的放入养老区,都满了就报OOM!

真理:经过研究,99%的对象都是临时对象,直接被清理!

老年区

新生区剩下来的,轻GC杀不死的。

永久区

这个区域常驻内存,用来存放JDK自身携带的Class对象,interface元数据,存储的是java运行时一些环境和类信息,该区域不存在垃圾回收GC。关闭虚拟机就会自动释放这个内存。

  • jdk1.6之前:永久代,常量池在方法区
  • jdk1.7:永久代,但是慢慢退化了(去永久代),常量池在堆中
  • jdk1.8之后:无永久代,常量池在元空间

一个启动类,加载了大量第三方jar包,Tomcat部署了太多应用,大量动态生成的反射类。不断被加载。直到内存满,就会出现OOM。

img

方法区又称为非堆,本质还是堆,只是为了区分概念。

元空间逻辑上存在,物理上不存在。

img

堆内存调优

报OOM怎么办?

1、尝试扩大堆内存,如果还报错,说明有死循环代码或者垃圾代码。

2、分析内存,看一下哪个地方有问题(专业工具)

使用Jprofiler工具分析OOM原因

在一个项目中,突然出现了OOM的故障,该如何排除?研究为什么会出错?

  • 能够看到代码第几行出错:内存快照分析工具,MAT,Jprofiler
  • Debug,一行行分析代码!

MAT,Jprofiler作用:

  • 分析Dump内存文件,快速定位内存泄漏
  • 获得堆中的数据
  • 获得大的对象(大厂面试)

看如下代码:

package com.linghu;

import java.util.ArrayList;

/**
 * @author linghu
 * @date 2024/1/24 10:44
 */
public class Demo03 {
    byte[] array = new byte[1*1024*1024]; //1m

    public static void main(String[] args) {
        ArrayList<Demo03> list = new ArrayList<>();
        int count = 0;

        try {
            while (true){
                list.add(new Demo03()); //不停地把创建对象放进列表,这是问题所在!
                count = count + 1;
            }
        } catch (Exception e) {
            System.out.println("count: "+count);
            e.printStackTrace();
        }

    }
}

报错如下

img

这个时候我们用Jprofiler工具分析OOM:

img

imgGC:垃圾回收

GC作用区域如下:

img

JVM在进行GC的时候,并不是对三个区域统一回收,大部分回收的是新生代。

  • 新生代
  • 幸存区
  • 老年区

GC分两种:

  • 轻GC
  • 重GC

关于GC面试题:

  • JVM的内存模型和分区~详细到每个分区放什么?
  • 堆里面的分区有哪些?Eden, from, to, 老年区,说说它们的特点!
  • GC算法有哪些?怎么用的?标记清除法,标记整理,复制算法,分代收集法。引用计数法。
  • 轻GC与重GC分别在什么时候发生?

GC算法

引用计数法

一般JVM不用,大型项目对象太多了!

img

引用次数为0的就会被清理掉!

img

复制算法

img

img

  • 好处:没有内存的碎片,内存效率高。
  • 坏处:浪费了内存空间(一个幸存空间永远是空的);假设对象100%存活,复制成本很高。

复制算法最佳使用场景:对象存活度较低的时候,新生区。

标记清除算法
标记清除

img

  • 优点:不需要额外空间,优化了复制算法
  • 缺点:两次扫描,严重浪费时间,会产生内存碎片
标记清除压缩

三部曲:

  • 标记
  • 清除
  • 压缩

img

总结

  • 内存效率:复制算法>标记清除算法>标记压缩算法(时间复杂度)
  • 内存整齐度:复制算法=标记压缩算法>标记清除算法
  • 内存利用率:标记压缩算法=标记清除算法>复制算法

难道没有最优算法吗?

答案:无,没有最好的算法,只有合适的算法(GC也被称为分代收集算法)。

  • 年轻代:存活率低,用复制算法。

  • 老年代:存活率高,区域大,用标记-清除-压缩。

    参考和研究:《深入理解Java虚拟机》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/347011.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

js执行字符串代码

js将字符串作为代码执行 一、适用场景二、具体实现1. eval2. new Function() 三、两者差异 一、适用场景 在业务中我们很少去将一个字符串作为代码执行&#xff0c;因为出于安全考虑&#xff0c;尽量不要直接在客户端执行用户输入的代码。但是在造轮子或者框架开发中&#xff…

鸿蒙常用UI效果及一些处理方式总结

前言&#xff1a; DevEco Studio版本&#xff1a;4.0.0.600 详细使用介绍 1、Text的一些常用设置 Text(this.message).fontSize(50)//字体大小.fontColor(Color.White)//字体颜色.fontWeight(FontWeight.Bold)//字体加粗.backgroundColor(Color.Black)//背景颜色.fontStyle(…

将Python打包为exe+inno setup将exe程序封装成向导安装程序

为什么要打包&#xff1f; Python脚本不能在没有安装Python的机器上运行。如果写了一个脚本&#xff0c;想分享给其他人使用&#xff0c;可她电脑又没有装Python。如果将脚本打包成exe文件&#xff0c;即使她的电脑上没有安装Python解释器&#xff0c;这个exe程序也能在上面运行…

简单易懂带你入门Spring框架,循序渐进,帮助你理解IOC思想---(一)入门实验一,教你如何使用spring框架

目录 1.1 、Spring概述 1.2 、Spring家族 1.3 、Spring Framework 1.3.1 、Spring Framework特性 1.3.2 、Spring Framework五大功能模块 IOC容器 IOC思想 IOC容器在Spring中的实现 基于XML管理bean 实验一&#xff1a;入门案例 如何获取bean 1.1 、Spring概述 官网…

爬虫笔记(二):实战58二手房

第一&#xff1a;给大家推荐一个爬虫的网课哈&#xff0c;码起来 第二&#xff1a;今夜主题&#xff1a;通过xpath爬取58二手房的title信息&#xff0c;也就是标红的位置~ 第三&#xff1a;先分析一波title所在的位置 打开按下f12打开抓包工具&#xff0c;即可看到网站的源码…

代码随想录算法训练营29期|day30 任务以及具体安排

332.重新安排行程 class Solution {private LinkedList<String> res;private LinkedList<String> path new LinkedList<>();public List<String> findItinerary(List<List<String>> tickets) {Collections.sort(tickets, (a, b) -> a.…

【必剪】拼字怎么使用?哪些素材能使用拼字?

1-1 点击需要编辑文字&#xff0c;在给定的人物中选择 其中一个&#xff0c;选择【拼字】选项&#xff0c;以下的字母表 的字母读音&#xff0c;是当前人物在他的鬼畜原版里的 全部读音&#xff0c;你可以打出自己专属独特的语句 ! 1-2 在字母表中继续寻找想要的读音&#xff…

利用STM32CubeMX和Keil模拟器,3天入门FreeRTOS(4.0) —— 动态创建队列

前言 &#xff08;1&#xff09;FreeRTOS是我一天过完的&#xff0c;由此回忆并且记录一下。个人认为&#xff0c;如果只是入门&#xff0c;利用STM32CubeMX是一个非常好的选择。学习完本系列课程之后&#xff0c;再去学习网上的一些其他课程也许会简单很多。 &#xff08;2&am…

基于YOLOv8的摔倒行为检测系统(Python源码+Pyqt6界面+数据集)

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文主要内容:通过实战基于YOLOv8的摔倒行为检测算法&#xff0c;从数据集制作到模型训练&#xff0c;最后设计成为检测UI界面 人体行为分析AI算法&#xff0c;是一种利用人工智能技术对人体行为进行检测、跟踪和分析的方法。通过计算…

MATLAB知识点:mode :计算众数

​讲解视频&#xff1a;可以在bilibili搜索《MATLAB教程新手入门篇——数学建模清风主讲》。​ MATLAB教程新手入门篇&#xff08;数学建模清风主讲&#xff0c;适合零基础同学观看&#xff09;_哔哩哔哩_bilibili 节选自第3章 3.4.1节 mode &#xff1a;计算众数 众数是指一…

log4j2配置文件命名及优先级

log4j 2.x版本不再支持像1.x中的.properties后缀的文件配置方式&#xff0c;2.x版本配置文件后缀名只能为".xml",“.json"或者”.jsn"。 命名规则 默认配置文件名&#xff1a; log4j2.xml 或 log4j2.json 测试或特定环境配置文件名&#xff1a;可以以 -t…

(blender学习)blender和vscode联合编写代码

文章目录 一、建立联合二、代码1.做一个小panel2. 给选定的物体更换好多好多材质 一、建立联合 按照这个文档做&#xff1a; 【Blender】使用 Microsoft Visual Studio Code 作为外部 IDE 来编写 Blender 脚本/附加组件 二、代码 下面几个代码的学习视频来自&#xff1a;ht…

Linux中并发程序设计(进程的创建和回收、exec函数使用)

进程的创建和回收 进程概念 概念 程序 存放在磁盘上的指令和数据的有序集合&#xff08;文件&#xff09; 静态的 进程 执行一个程序所分配的资源的总称 动态的进程和程序比较 注&#xff1a;进程是存在RAM中&#xff0c;程序是存放在ROM(flash)中的进程内容 BSS段&#xff…

Rocky8 顺利安装 Airflow 并解决数据库报错问题

rocky是替代centos的服务器系统&#xff0c;稳定可靠。rocky8会比centos7新&#xff0c;可以支持更多服务软件的安装&#xff0c;免去升级各种库的麻烦&#xff0c;本文运行airflow服务就用rocky8系统。airflow是一个定时任务管理系统&#xff0c;功能强大&#xff0c;目前是ap…

【QT+QGIS跨平台编译】之七:【libjpeg+Qt跨平台编译】(一套代码、一套框架,跨平台编译)

文章目录 一、libjpeg介绍二、文件下载三、文件分析四、pro文件五、编译实践一、libjpeg介绍 libjpeg是一个广泛使用的jpeg图像压缩和解压的函数库,采用 C 语言开发。 2013年1月,Independent JPEG Group发布了版本9,对新引入的无损编码模式进行了改进。2022年1月,发布了版…

Android U 配置 WiFiCalling 场景下PLMN/SPN 显示的代码逻辑介绍

功能介绍 根据设备的网络连接情况更新状态栏显示的运营商及网络状态。 注册上WFC&#xff08;WiFi Calling&#xff09;后&#xff0c;支持客制化显示左上角状态栏中的运营商网络状态信息 。具体的代码逻辑在CarrierDisplayNameResolver.java。 ServiceStateTracker 网络状态…

C# Bitmap类学习1

Bitmap对象封装了GDI中的一个位图&#xff0c;此位图由图形图像及其属性的像素数据组成.因此Bitmap是用于处理由像素数据定义的图像的对象。 using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using …

程序媛的mac修炼手册-- 如何用Python节省WPS会员费

上篇分享了如何用微博爬虫&#xff0c;咱举例爬了女明星江疏影的微博数据。今天就用这些数据&#xff0c;给大家安利一下怎么用Python实现WPS中部分Excel付费功能。 MacOS系统自带的工具&#xff0c;绝大多数都非常顶&#xff0c;除Numbers外。当然&#xff0c;page比起word来&…

图像分类】【深度学习】【轻量级网络】【Pytorch版本】EfficientNet_V2模型算法详解

【图像分类】【深度学习】【轻量级网络】【Pytorch版本】EfficientNet_V2模型算法详解 文章目录 【图像分类】【深度学习】【轻量级网络】【Pytorch版本】EfficientNet_V2模型算法详解前言EfficientNet_V2讲解自适应正则化的渐进学习(Progressive Learning with adaptive Regul…

魔众题库系统v9.3.0版本:升级与新功能亮点,让学习更高效!

大家好&#xff01;今天我们激动地向大家宣布&#xff0c;魔众题库系统已经升级到了v9.3.0版本&#xff01;这个版本带来了许多令人兴奋的改进和新功能&#xff0c;让用户的使用体验更上一层楼。 首先&#xff0c;让我们来看看这个版本的VIP界面升级。无论是PC端还是移动端&am…