当前位置: 首页 > article >正文

关于 JVM 个人 NOTE

目录

1、JVM 的体系结构

2、双亲委派机制

3、堆内存调优

4、关于GC垃圾回收机制

4.1 GC中的复制算法

4.2 GC中的标记清除算法


1、JVM 的体系结构

"堆"中存在垃圾而"栈"中不存在垃圾的原因:

堆(Heap)

  1. 用途:堆主要用于存储对象实例和数组。在Java中,几乎所有通过new关键字创建的对象都会存储在堆内存中。
  2. 内存分配与释放堆内存的分配和释放由JVM的垃圾回收器(Garbage Collector, GC)自动管理。当对象不再被引用时,它们被认为是垃圾,但并不会立即被释放。相反,垃圾回收器会在合适的时机进行垃圾回收,以释放这些不再使用的内存空间。
  3. 垃圾存在的原因:由于垃圾回收器并不是实时运行的,且对象的生命周期可能跨越多个垃圾回收周期,因此在堆内存中可能会存在已经不被使用但仍未被回收的垃圾对象

栈(Stack)

  1. 用途:栈主要用于存储局部变量、方法调用信息和对象的引用变量(但对象本身存储在堆中)。每个线程都有自己独立的栈空间,用于存储该线程执行过程中的局部变量和方法调用信息。
  2. 内存分配与释放栈内存的分配和释放是自动的,且与方法的调用和返回紧密相关。每当一个方法被调用时,JVM会在栈上为该方法创建一个新的栈帧(Stack Frame),用于存储局部变量等信息。当方法执行完毕后,其对应的栈帧会自动从栈中弹出并销毁,局部变量占用的内存也会随之释放。
  3. 不存在垃圾的原因:由于栈内存的分配和释放都是自动的,且与方法的调用和返回紧密相关,因此栈内存中的局部变量和方法调用信息在不再需要时会自动被释放,不会出现像堆内存中那样的垃圾对象。此外,栈内存的生命周期与线程相同,当线程结束时,其栈内存也会被自动释放


2、双亲委派机制

一开始先从Java启动类加载器中寻找相关类,若没有,则在拓展类加载器中寻找,最后才会在当前应用类加载器中寻找

例:如图所示

这里创建了一个java.lang的包,定义了String以及toString,与Java自带包冲突;

说明了一开始是由Java启动类加载器(Boot)进行类的加载,发现具有相同重复的类,则抛出异常,主要是为了保证安全性,防止恶意的擅自更改善意的代码

报错信息:


3、堆内存调优

public static void main(String[] args) {
        
        //jvm试图使用的最大内存
        long max = Runtime.getRuntime().maxMemory();

        //jvm总内存
        long total = Runtime.getRuntime().totalMemory();

        System.out.println("max="+max+"字节\t"+(max/(double)1024/1024)+"MB");
        System.out.println("total="+total+"字节\t"+(total/(double)1024/1024)+"MB");
    }

这里进行输出JVM中默认配置下的相关内存大小:

存在问题:

若发生OOM内存溢出,默认的堆配置内存可能行不通,可能需要自己进行配置

点击编辑配置        --->        修改选项        --->        添加VM选项

这里进行输入的配置中,其中“1024m”表示内存的大小、“PrintGCDetails”表示打印相关信息(更多的参数去百度),可以自行根据情况调节,以尽量控制不会出现OOM内存溢出

输出结果:

可见,通过配置,堆中所分配的内存大小发生了改变


4、关于GC垃圾回收机制

GC的作用区域在 “堆内存区” 以及 “方法区”

4.1 GC中的复制算法

内部具体伪动态图:

由于谁空谁为To区,Eden以及From中的对象都会往To区集合;当集合完毕之后,原来的From区为空,所以变为了To区,与原来的To区进行了“身份交换”;而后续存活下来的对象则顺利进入“养老区”

优点:没有内存的碎片,即 “要走一块走,要留一块留”

缺点:浪费了内存空间,即 “To幸存区” 与 “From幸存区” 必须空一块

总结

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

4.2 GC中的标记清除算法

优点:不需要占用额外的空间

缺点:经过两次扫描,严重浪费空间,因为会产生内存碎片

对于标记清除算法,对应的优化算法还有 “标记压缩算法” 、“标记清除压缩算法”,但都存在利与弊,没有最优,只有最合适


http://www.kler.cn/a/327183.html

相关文章:

  • ARM Assembly: 第8课 branching
  • Web自动化中常用XPath定位方式
  • D23【 python 接口自动化学习】- python 基础之判断与循环
  • Docker入门指南:快速学习Docker的基本操作
  • 网络编程(13)——单例模式
  • BCJR算法——卷积码的最大后验译码
  • Ubuntu 开机自启动 .py / .sh 脚本,可通过脚本启动 roslaunch/roscore等
  • 联邦学习(三只决策和大数据分析)(学习笔记)
  • 【网络安全】TCP和UDP
  • 防止电脑电池老化,禁止usb或者ac接口调试时充电
  • 计算神经学笔记01
  • 后端-对表格数据进行添加、删除和修改
  • 单片机的原理及应用
  • 2024年华为OD机试真题-找终点-Java-OD统一考试(E卷)
  • AIGC学习笔记—minimind详解+训练+推理
  • elasticsearch单个node节点写入数据
  • 中间层架构设计:构建稳健的企业级服务
  • [Day 81] 區塊鏈與人工智能的聯動應用:理論、技術與實踐
  • 表现层架构设计:打造高效、可维护的前端系统
  • JavaScript网页设计案例深度解析:从理论到实践