JVM知识总结(简单且高效)

1. JVM内存与本地内存

  • JVM内存:受虚拟机内存大小的参数控制,当大小超过参数设置的大小时会报OOM。
  • 本地内存:本地内存不受虚拟机内存参数的限制,只受物理内存容量的限制;虽然不受参数的限制,如果所占内存超过物理内存,仍然会报OOM。

2. JVM内存结构

  • 虚拟机栈:服务于Java方法
  • 本地方法栈:服务于本地方法
  • 程序计数器:保存当前线程执行的字节码位置,当然每个线程工作时都有独立的计数器。
  • 堆:用于存放对象
  • 方法区:用于存放常量、静态变量、数据类型、类信息等元数据
    在这里插入图片描述

3. 线程独占与共享区域

  • 独占:虚拟机栈、本地方法栈、程序计数器
  • 共享:堆、方法区

4. 栈与堆的区别

  • 栈:是运行时单位,代表逻辑,且区域连续
  • 堆:是存储单位,代表数据,区域不连续

5. 方法区、永久代、元空间

  • 方法区:是规范,是概念。
  • 永久代:是实现,在JDK7及之前版本,是方法区的一种实现。
  • 元空间:是实现,在JDK8及之后版本,是方法区的一种实现。(替代了永久代)
    在这里插入图片描述
    在这里插入图片描述

6. 元空间为什么替代了永久代

  • 突破内存限制,减少OOM。 由于元空间使用的是本地内存,而不是 JVM 内存。
  • 提高 Full GC 的效率。 因为永久代中存放了很多 JVM 需要的类信息,这些数据大多数是不会被清理的,所以 Full GC 往往无法回收多少空间。但在元空间模型中,由于字符串常量池已移至堆中,因此可以更有效地进行垃圾回收,避免了因频繁的 Full GC 导致的性能影响。
  • 满足不同的类加载需求和动态类加载的情况。 元空间可以动态地调整大小,在一些大型的、模块化的应用中,可能需要加载大量的类,这就需要大量的元数据存储空间。
  • 避免永久代调优和大小设置的复杂性。 在 Java8 之前的版本中,通常需要手动设置永久代的大小,以避免内存溢出的错误。这增加了应用的配置和管理的复杂性。而元空间使用本地内存,根据实际需求动态调整,大大简化了内存管理的复杂性。

7. JVM内存可见性

  • 线程对于变量的操作只能在自己的工作内存中进行,而不能直接对主存操作。
    在这里插入图片描述

8. 类的生命周期

  • 加载:通过类的完全限定名,查找此类字节码文件,并创建Class对象。
  • 验证:确保Class文件符合当前虚拟机的要求。
  • 准备:为static修饰的类变量分配内存,不包含final修饰的变量,因为final已经在编译时分配。
  • 解析:将常量池的符号引用替换为直接引用。
  • 初始化:静态块执行、静态变量赋值。
  • 使用:new出对象程序中使用。
  • 卸载:执行垃圾回收。
    在这里插入图片描述

9. 符号引用和直接引用

  • 符号引用即用**(用字符串符号的形式)**来表示引用,其实被引用的类、方法或者变量还没有被加载到内存中。而直接引用则是有具体引用地址的指针,被引用的类、方法或者变量已经被加载到内存中。

10. 类的初始化

只有对类主动使用时才会初始化,触发条件包括

  • 创建类的实例时。
  • 访问类的静态方法或静态变量的时候。
  • 使用Class.forName反射类的时候。
  • 某个子类初始化的时候。

11. 双亲委派加载机制

  • AppClassLoader从缓存查找类,没有则委托给父加载器ExtClassLoader
  • ExtClassLoader从缓存查找类,没有则委托给父加载器BootStrapClassLoader
  • BootStrapClassLoader从缓存查找类,没有则sun.mic.boot.class路径查找
  • BootStrapClassLoadersun.mic.boot.class路径查找,没有则让子类ExtClassLoader加载
  • ExtClassLoaderjava.ext.dirs路径查找,没有则让子类AppClassLoader加载
  • AppClassLoaderjava.class.path路径查找,如果找到就加载类,否则就抛出异常。
    在这里插入图片描述

12. 双亲委派机制的优点

  • 避免类的重复加载
  • 避免Java的核心API被篡改

13. GC如何判断对象可以被回收

  • 引用计数法:每个对象有一个引用计数属性,新增一个引用时计数加1,引用释放时计数减1,计数为0时可以回收
  • 可达性分析法:从GC Roots开始向下搜索,搜索所走过的路径称为引用链,当一个对象到GC Roots没有任何引用链相连时,则证明此对象是不可用的,那么虚拟机就判断是可回收对象

14. 可达性分析算法

  • 可达性算法中的不可达对象并不是立即死亡的,对象拥有一次自我拯救的机会
  • 当对象编程(GC Roots)不可达时,GC会判断该对象是否执行过finalize方法,若执行了则直接回收。否则,若对象未执行过finalized方法,将其放入F-Queue队列。执行finalize方法完毕后,GC会再次判断该对象是否可达,若不可达,则进行回收,否则对象复活。

15. 四种JVM的垃圾回收算法

标记-清除算法:标记无用对象,然后进行清除回收。缺点:效率不高,无法清除垃圾碎片。
标记-整理算法:标记无用对象,让所有存活的对象都向一端移动,然后直接清除掉端边界以外的内存。
复制算法:按照容量划分二个大小相等的内存区域,当一块用完的时候将活着的对象复制到另一块上,然后再把已使用的内存空间一次清理掉。缺点:内存使用率不高,只有原来的一半。
分代算法:根据对象存活周期的不同将内存划分为几块,一般是新生代和老年代,新生代基本采用复制算法,老年代采用标记整理算法。

16. 分代算法

  1. new新对象放入堆中
  2. 如果在Eden区没有空间,那就发生Minor GC,保留存活的对象
  3. 当新生代仍然没有空间,那就将部分年龄大的新生代对象放入老年代
  4. 当老年代仍然没有空间,那就发生Major GC,保留存活的对象
  5. Major GC之后仍然没有空间,就会抛出OOM
    在这里插入图片描述

17. 为什么会出现OOM

  • 内存泄漏:申请使用完的内存没有释放,导致虚拟机不能再次使用该内存,此时这段内存就泄露了。因为申请者不用了,而又不能被虚拟机分配给别人用
  • 内存溢出:申请的内存超出了 JVM 能提供的内存大小,此时称之为溢出内存泄漏持续存在,最后一定会溢出,两者是因果关系

18. 常见的OOM报错

方法区溢出

  • 报错信息:java.lang.OutOfMemoryError: PermGen space
  • 常见问题:一般出现于大量Class对象,或者方法区过小

栈区溢出

  • 报错信息:java.lang.StackOverflowError
  • 常见问题:一般是由于程序中存在 死循环或者深度递归调用 造成的,或者栈区过小

堆区溢出

  • 报错信息:java.lang.OutOfMemoryError: Java heap space
  • 常见问题:内存泄漏、内存溢出

19. OOM如何解决

  • 主要使用dump文件jprofiler两个工具,具体自行学习

20. JVM的元空间中会发生垃圾回收吗

  • 垃圾回收不会发生在元空间,如果元空间满了或者是超过了临界值,会触发完全垃圾回收(FullGC)。
  • 如果正确设置元空间大小,那么就可以避免Full GC

21. 查看相关线程命令

  • jps(JVM Process Status Tool):显示指定系统内所有的HotSpot虚拟机进程。
  • jstat(JVM statistics Monitoring):显示出类装载、内存、垃圾收集、JIT编译等运行数据。
  • jmap(JVM Memory Map):命令用于生成heap dump文件
  • jhat(JVM Heap Analysis Tool):命令是与jmap搭配使用,用来分析jmap生成的dump,jhat内
    置了一个微型的HTTP/HTML服务器,生成dump的分析结果后,可以在浏览器中查看
  • jstack:用于生成java虚拟机当前时刻的线程快照。
  • jinfo(JVM Configuration info):这个命令作用是实时查看和调整虚拟机运行参数。

22. Minor GC与Full GC分别在什么时候发生?

  • 新生代内存不够用时候发生MGC也叫YGC,JVM内存不够的时候发生FGC。

23. 对象一定分配在堆中吗?有没有了解逃逸分析技术?

  • 不一定的,JVM通过「逃逸分析」,那些逃不出方法的对象会在栈上分配。

24. 什么是逃逸分析

  • 方法逃逸:在一个方法体内,定义一个局部变量,而它可能被外部方法引用,比如作为调用参数传递给方法,或作为对象直接返回。或者,可以理解成对象跳出了方法。
  • 线程逃逸:这个对象被其他线程访问到,比如赋值给了实例变量,并被其他线程访问到了。对象逃出了当前线程。
  • 逃逸分析:没有发生逃逸的对象,会进一步优化,将其放在栈中
// sb就逃逸了
public static StringBuffer craeteStringBuffer(String s1, String s2) {StringBuffer sb = new StringBuffer();sb.append(s1);sb.append(s2);return sb;
}// sb没有逃逸了
public static String createStringBuffer(String s1, String s2) {StringBuffer sb = new StringBuffer();sb.append(s1);sb.append(s2);return sb.toString();
}

25. 逃逸分析带来的好处

  • 同步省略:如果对象只能一个线程被访问到,那么对于这个对象的操作可以不考虑同步。
  • 将堆分配转化为栈分配:对象可能是栈分配的候选,而不是堆分配。
  • 分离对象:有的对象可以,那么对象的部分(或全部)可以不存储在内存,而是存储在CPU寄存器中。

26. 逃逸分析的开启关闭

  • 从jdk 1.7开始已经默认开始逃逸分析,如需关闭,需要指定-XX:-DoEscapeAnalysis
  • -XX:+DoEscapeAnalysis : 表示开启逃逸分析
  • -XX:-DoEscapeAnalysis : 表示关闭逃逸分析

27. 同步省略
动态编译同步块的时候,JIT编译器可以借助逃逸分析来证实同步块的对象只能够被一个线程访问,就会取消对这部分代码的同步,这个过程也叫锁消除

如以下代码:

public void f() {Object hollis = new Object();synchronized(hollis) {System.out.println(hollis);}
}

hollis对象的生命周期只在f()方法中,并不会被其他线程所访问到,所以在JIT编译阶段优化成:

public void f() {Object hollis = new Object();System.out.println(hollis);
}

28. 元空间为什么能替代永久代

  • 区别:永久代存在于JVM限制内存,元空间存在于本地内存。所以元空间会更大。
  • 好处:能够避免方法区OOM异常,虽然我们可以通过设置永久代的大小来解决OOM,但是不是总能知道应该设置为多大合适, 如果使用默认值很容易遇到OOM错误。当使用元空间时,可以加载多少类的元数据就由系统的实际可用空间来控制啦,即元空间具有伸缩性。

29. Stop The World

  • 进行垃圾回收的过程中,会涉及对象的移动。为了保证对象引用更新的正确性,必须暂停所有的用户线程,像这样的停顿,虚拟机设计者形象描述为「Stop The World」,也简称STW。

30. 指针碰撞

  • 分配方式:一般情况下,JVM的对象都放在堆内存中(发生逃逸分析除外)。当类加载检查通过后,Java虚拟机开始为新生对象分配内存。如果Java堆中内存是绝对规整的,所有被使用过的的内存都被放到一边,空闲的内存放到另外一边,中间放着一个指针作为分界点的指示器,所分配内存仅仅是把那个指针向空闲空间方向挪动一段与对象大小相等的实例。
  • 指针碰撞:如果现在两个线程同时创建对象呢?
    在这里插入图片描述

31. 空闲列表

  • 如果Java堆内存中的内存并不是规整的,已被使用的内存和空闲的内存相互交错在一起,不可以进行指针碰撞啦,虚拟机必须维护一个列表,记录哪些内存是可用的,在分配的时候从列表找到一块大的空间分配给对象实例,并更新列表上的记录,这种分配方式就是空闲列表。

32. 什么是TLAB

  • 每个线程都有自己的一小块内存,这就是TLAB(Thread Local Allocation Buffer,本地线程分配缓存) 。虚拟机通过 -XX:UseTLAB 设定它的。

33. JVM有哪些垃圾回收器

  • Serial:新生代、单线程、复制算法(适用于小堆、单核)
  • ParScavenge:新生代、多线程、复制算法、高吞吐
  • Serial Old:老年代、单线程、标记-整理(适用于大堆、多核)
  • ParOld:老年代、多线程、标记-整理、低交互
  • CMS:老年代、多线程、标记-整理、低停顿低吞吐
  • G1:全堆、多线程、标记-整理

附录

「堆栈内存相关」

-Xms 设置初始堆的大小
-Xmx 设置最大堆的大小
-Xmn 设置年轻代大小,相当于同时配置-XX:NewSize和-XX:MaxNewSize为一样的值
-Xss 每个线程的堆栈大小
-XX:NewSize 设置年轻代大小(for 1.3/1.4)
-XX:MaxNewSize 年轻代最大值(for 1.3/1.4)
-XX:NewRatio 年轻代与年老代的比值(除去持久代)
-XX:SurvivorRatio Eden区与Survivor区的的比值
-XX:PretenureSizeThreshold 当创建的对象超过指定大小时,直接把对象分配在老年代。
-XX:MaxTenuringThreshold设定对象在Survivor复制的最大年龄阈值,超过阈值转移到
老年代

「垃圾收集器相关」

-XX:+UseParallelGC:选择垃圾收集器为并行收集器。
-XX:ParallelGCThreads=20:配置并行收集器的线程数
-XX:+UseConcMarkSweepGC:设置年老代为并发收集。
-XX:CMSFullGCsBeforeCompaction=5 由于并发收集器不对内存空间进行压缩、整理,
所以运行一段时间以后会产生“碎片”,使得运行效率降低。此值设置运行5次GC以后对内
存空间进行压缩、整理。
-XX:+UseCMSCompactAtFullCollection:打开对年老代的压缩。可能会影响性能,但是
可以消除碎片

「辅助信息相关」

-XX:+PrintGCDetails 打印GC详细信息
-XX:+HeapDumpOnOutOfMemoryError让JVM在发生内存溢出的时候自动生成内存快照,
排查问题用
-XX:+DisableExplicitGC禁止系统System.gc(),防止手动误触发FGC造成问题.
-XX:+PrintTLAB 查看TLAB空间的使用情况

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

软件工程概论------文件管理

目录 1.文件的相关概念 2.文件目录 3.位示图 4.索引文件 5.例题 1.文件的相关概念 文件:具有符号名的、在逻辑上具有完整意义的一组相关信息项的集合。 逻辑结构:有结构的记录式文件、无结构的流式文件。 物理结构: 连续结构、链接结构、索引结构、多个物理块的索引表。 …

ROS-arbotix安装

方式一:命令行输入: sudo apt-get install ros-melodic-arbotix如果ROS为其他版本,可将melodic替换为对应版本。 方式二: 先从 github 下载源码,然后调用 catkin_make 编译 git clone https://github.com/vanadiumla…

MySQL--基础篇

这里写目录标题 总览MySQl各个阶段基础篇总览 MySQL概述数据库相关概念查看本机MySQL版本号启停mysql打开windows服务管理windows命令行启停 连接mysql客户端mysql运行逻辑数据模型关系型数据库 总结 SQL总览SQL通用语法SQL语句分类DDL数据库操作表操作查询表创建表结构数据类型…

sublim安装Autoprefixer插件

有时候在写css样式的时候,分不清哪些属性需要前缀,哪些不需要写前缀,sublime text这款编辑器下安装autoprefixer这款插件可以省去很多问题,写起来也很方便。1 确保系统已经安装node.js 可直接去官网上下载并安装,我的系…

c语言:用结构体找出学生年龄|练习题

一、题目 在结构体数组中&#xff0c;输入学生信息&#xff0c;找出学生的年龄。 如图&#xff1a; 二、代码图片【带注释】 三、源代码【带注释】 #include <stdio.h> //设置结构体&#xff0c;结构体有3个变量 struct student { int id; char name[20]; …

BMTrain来高效训练预训练模型-大模型的福音

一.背景知识 在2018年&#xff0c;预训练语言模型技术的出现成为人工智能领域一场革命性的变革。研究表明&#xff0c;通过增加模型参数量和训练数据规模&#xff0c;可以有效提升语言模型的性能&#xff0c;因此十亿、百亿甚至千亿级大模型的探索成为业界的热门话题。这一趋势…

使用Spring Cache优化数据库访问

使用Spring Cache优化数据库访问 在这篇博客中&#xff0c;我们将学习如何使用Spring Cache来优化数据库访问&#xff0c;提高系统性能。我们将创建一个简单的图书管理应用作为示例&#xff0c;并演示如何通过缓存减少对数据库的频繁查询。 1. 项目结构 首先&#xff0c;我们…

使用vite构建Vue3项目

1、安装vite npm init vitelatest npm构建vite项目 yarn create vite yarn构建vite项目2、依次需要配置项目名 、框架选择、原生和ts版本的选择 r enter 重新开始服务 o enter 快速打开浏览器3、项目启动效果

TCP_可靠数据传输原理

引言 在网络通信中&#xff0c;TCP是确保数据可靠传输的关键协议。但在我们深入研究TCP拥塞控制技术之前&#xff0c;让我们先探索可靠数据传输的原理&#xff0c;特别是TCP头部中一些重要字段的作用。 网络层提供了点对点的通信服务&#xff0c;努力交付数据报&#xff0c;但…

Python课程设计基于python的人脸识别佩戴口罩系统设计

wx供重浩&#xff1a;创享日记 对话框发送&#xff1a;python口罩 获取完整论文报告源码源文件 1 研究背景与意义 新型冠状病毒展现出全球化流行和蔓延的趋势&#xff0c;这提醒我们&#xff1a;传染病防治在今后相当长时间内仍是疾病预测控制工作的重点。戴口罩是预防呼吸道…

Qt应用-实现图像截取功能类似QQ上传头像截取功能

本文演示利用Qt实现图像截取功能类似QQ上传头像截取功能。 效果如下,通过移动中间的裁剪区域可以获得一张裁剪后的图片。 目录

UE4运用C++和框架开发坦克大战教程笔记(十四)(第43~45集)

UE4运用C和框架开发坦克大战教程笔记&#xff08;十四&#xff09;&#xff08;第43~45集&#xff09; 43. 单个加载 UObject 功能获取资源 URL 链接实现异步加载单个 UObject 类型资源 44. 批量加载 UObject 功能测试加载单个 UObject 资源批量加载多个同类的 UObject 资源 45…

c# OpenCvSharp透视矫正参数调整器

透视矫正不够智能化&#xff0c;每次都要进行局部参数调整&#xff0c;不便于程序使用&#xff0c;程序流程还是那几个步骤&#xff1b; 1、读取图像、灰度化 2、高斯滤波 3、二值化 4、边缘检测 灰度化图 上个图看看经过调整透视矫正边缘检测结果我还是挺满意的 发现一个…

MongoDB数据类型详解

BSON 协议与数据类型 MongoDB 为什么会使用 BSON&#xff1f; JSON 是当今非常通用的一种跨语言 Web 数据交互格式&#xff0c;属 ECMAScript 标准规范的一个子集。JSON &#xff08;JavaScript Object Notation&#xff0c;JS 对象简谱&#xff09;即 JavaScript 对象表示法…

开源项目 | 完整部署流程、一款开源人人可用的开源数据可视化分析工具

&#x1f4da; 项目介绍 在互联网数据大爆炸的这几年&#xff0c;各类数据处理、数据可视化的需求使得 GitHub 上诞生了一大批高质量的 BI 工具。 借助这些 BI 工具&#xff0c;我们能够大幅提升数据分析效率、生成更高质量的项目报告&#xff0c;让用户通过直观的数据看到结…

微服务实战系列之API加密

前言 随着一阵阵凛冽寒风的呼啸&#xff0c;新的年轮不知不觉滚滚而来。故事随着2023的远去&#xff0c;尘封于案底&#xff1b;希望迎着新年&#xff0c;绽放于枝头。在2024新岁启航&#xff0c;扬帆破浪之时&#xff0c;让烦恼抛洒于九霄&#xff0c;让生机蓬勃于朝朝暮暮。 …

如何充值GPT会员账号?

详情点击链接&#xff1a;如何充值GPT会员账号&#xff1f; 一OpenAI 1.最新大模型GPT-4 Turbo 2.最新发布的高级数据分析&#xff0c;AI画图&#xff0c;图像识别&#xff0c;文档API 3.GPT Store 4.从0到1创建自己的GPT应用 5. 模型Gemini以及大模型Claude2二定制自己的…

基于多反应堆的高并发服务器【C/C++/Reactor】(中)创建并初始化TcpServer实例 以及 启动

对于一个TcpServer来说&#xff0c;它的灵魂是什么&#xff1f;就是需要提供一个事件循环EventLop(EventLoop)&#xff0c;不停地去检测有没有客户端的连接到达&#xff0c;有没有客户端给服务器发送数据&#xff0c;描述的这些动作&#xff0c;反应堆模型能够胜任。当服务器和…

【Docker】容器的相关命令

上一篇&#xff1a;创建&#xff0c;查看&#xff0c;进入容器 https://blog.csdn.net/m0_67930426/article/details/135430093?spm1001.2014.3001.5502 目录 1. 关闭容器 2.启动容器 3.删除容器 4.查看容器的信息 查看容器 1. 关闭容器 从图上来看&#xff0c;容器 aa…

机器学习-基于attention机制来实现对Image Caption图像描述实验

机器学习-基于attention机制来实现对Image Caption图像描述实验 实验目的 基于attention机制来实现对Image Caption图像描述 实验内容 1.了解一下RNN的Encoder-Decoder结构 在最原始的RNN结构中&#xff0c;输入序列和输出序列必须是严格等长的。但在机器翻译等任务中&…