一章带你了解Java虚拟机——JVM(揭秘java虚拟机:jvm设计原理与实现)

createh51个月前 (02-01)技术教程22


1、JVM 的体系结构


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

堆(Heap)

用途:堆主要用于存储对象实例和数组。在Java中,几乎所有通过new关键字创建的对象都会存储在堆内存中。

内存分配与释放:堆内存的分配和释放由JVM的垃圾回收器(Garbage Collector, GC)自动管理。当对象不再被引用时,它们被认为是垃圾,但并不会立即被释放。相反,垃圾回收器会在合适的时机进行垃圾回收,以释放这些不再使用的内存空间。

垃圾存在的原因:由于垃圾回收器并不是实时运行的,且对象的生命周期可能跨越多个垃圾回收周期,因此在堆内存中可能会存在已经不被使用但仍未被回收的垃圾对象

栈(Stack)

用途:栈主要用于存储局部变量、方法调用信息和对象的引用变量(但对象本身存储在堆中)。每个线程都有自己独立的栈空间,用于存储该线程执行过程中的局部变量和方法调用信息。

内存分配与释放:栈内存的分配和释放是自动的,且与方法的调用和返回紧密相关。每当一个方法被调用时,JVM会在栈上为该方法创建一个新的栈帧(Stack Frame),用于存储局部变量等信息。当方法执行完毕后,其对应的栈帧会自动从栈中弹出并销毁,局部变量占用的内存也会随之释放。

不存在垃圾的原因:由于栈内存的分配和释放都是自动的,且与方法的调用和返回紧密相关,因此栈内存中的局部变量和方法调用信息在不再需要时会自动被释放,不会出现像堆内存中那样的垃圾对象。此外,栈内存的生命周期与线程相同,当线程结束时,其栈内存也会被自动释放




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中的标记清除算法

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

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


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

相关文章

阿里P9级架构师马士兵花四小时让你搞懂JVM底层原理(附JVMpdf)

JVM面试难题当面试官向你提问JVM的实战问题时,你是否感觉到无所适从?想解决生产环境中的GC问题,你是否感觉到狗咬刺猬,无处下嘴?面对JVM的海量参数,你是否感觉到迷茫无助?看过很多的JVM文章,但...

Java虚拟机(jvm)-简介(深入浅出:java虚拟机设计与实现)

一、Java运行时虚拟机内存区域划分1.元空间(Metaspace)元空间(Metaspace)从java8开始替换掉了原来的方法区(Method Area)。相比方法区(Method Area)在元...

Java 动态调试技术原理及实践(java的动态性指的是什么)

调试是发现和减少计算机程序或电子仪器设备中程序错误的一个过程。最常用的断点调试技术会在断点位置停顿,导致应用停止响应。本文将介绍一种Java动态调试技术,希望能对大家有帮助。同时也欢迎读者朋友们一起交...

教你用Java开发一个简单的JVM(java开发实例教程)

一、前言几年前,接到一个开发任务:用Java开发能运行Java智能合约的虚拟机。在开发Java智能合约时,只能使用智能合约SDK提供的类和一些Java常用类(8种基本数据类型包装类;String、Bi...

Java的优势:跨平台只是一部分(java优劣势)

以下讨论只针对PC端和移动端。Java最大的优势真的在于跨平台吗?以前是,但现在已经不是了。有跨平台需求的仅仅是客户端应用,而不是服务端。例如桌面应用,你的客户可能是Windows用户,也可能是Lin...