JVM基础知识(内存区域划分,类加载,GC垃圾回收)

2023-09-21 17:17:08

目录

内存区域划分

JVM中的栈

JVM中的堆

程序计数器

方法区(元数据区)

给一段代码,某个变量在哪个区域上?

类加载

类加载时机

双亲委派模型

GC 垃圾回收机制

GC 实际工作过程

1.找到垃圾/判定垃圾

1.可达性分析(Java中的做法)

2.引用计数

2.清理垃圾

1.标记清除

2.复制算法

3.标记整理

分代回收(复制算法+标记整理)


内存区域划分

如果内存区域只有一块,不太方便,为了更加方便使用,就把整个空间隔成很多区域,每一个区域都有不同的作用
JVM,在启动的时候,会申请一整个很大的区域,JVM 是一个应用程序,从操作系统里申请内存,JVM把整个空间分层几个部分,每个部分各自有不同的功能作用
每一个Java进程都包含一个JVM

JVM中的栈

JVM中的栈不是数据结构中的栈,是JVM中的一个特定空间,对于 JVM 虚拟机,这里存储是 方法(我们自己写的java代码中的方法) 之间的调用关系.对于 本地方法栈,存储的是JVM内部方法的调用关系

整个栈空间内部,可以认为包含很多元素,每个元素表示一个方法. 这里的每个元素,称为一个"栈帧",这一个栈帧里,会包含这个方法的 入口地址,方法的参数是什么,返回地址是什么,局部变量等

数据结构的栈,是一个通用的更广泛的概念,是后进先出的数据结构,此处的JVM中的栈,特指JVM上的一块内存空间,由于函数调用,也是有后进先出的特点

JVM中虚拟机中的栈,有很多,每一个线程都有一个属于自己的栈,每一个栈都有很多的栈帧,调用一个方法会创建栈帧,方法结束,就会销毁这个栈帧

JVM中的堆

堆是整个 JVM 空间最大的区域,new 出来的对象(引用类型),都是在堆上.因此类的成员变量也在堆山.
堆是一个进程只有一份,一个进程中的多个线程共用一份堆.栈是一个线程有一个栈,一个进程有N个栈
堆的生命周期比较长,堆上面的方法执行结束默认不自动释放空间,而栈上面的方法会随着方法执行结束,自动释放空间

程序计数器

记录当前线程执行到哪个指令,每个线程都独有一份程序计数器

方法区(元数据区)

方法区每个进程只有一个,多个线程共用一份类对象,常量池,静态成员(static)都在方法区

给一段代码,某个变量在哪个区域上?

原则
1.局部变量在 栈 上
2.普通成员变量在 堆 上
3.静态成员变量在 方法区/元数据区 上

类加载

类加载: 类加载就是 .class文件,从文件(硬盘)被加载到内存中(方法区/元数据区)这样的过程

加载: 把.class文件找到,打开文件,读文件,把文件内容读到内存中,最终得到类对象
验证: 检查.class文件格式是否正确 
准备: 给类对象分配一个内存空间(在方法区/元数据区占个位置),会使静态成员被设置成0值
解析: 初始化字符串常量,把符号引用转换为直接引用
初始化: 调用构造方法,进行成员初始化,执行代码块,静态代码块,加载父类...      

类加载时机

java程序运行,不是把所有的类都加载了,而是真正用到了才加载(懒汉模式),一旦加载过后,后续再使用就不必重复加载了

1.构造类的实例
2.调用这个类的 静态方法/使用静态属性
3.加载子类,就会先加载父类

双亲委派模型

双亲委派模型,描述的是 加载过程 找.class文件,基本过程

JVM默认提供了 三个 类加载器

BootstrapClassLoader:  负责加载标准库中的类(java规范)

ExtensionClassLoader:  负责加载JVM扩展中的类(规范之外)

ApplicationClassLoader: 负责加载用户提供的第三方库/用户项目代码 中的类

上述三个类存在父子关系,BootstrapClassLoader是ExtensionClassLoader的父类,ExtensionClassLoader是ApplicationClassLoader的父类

加载一个类的时候是先从ApplicationClassLoader开始的,但是 ApplicationClassLoader会把加载任务,交给父亲,让父亲去执行.于是ExtensionClassLoader要去加载,但是ExtensionClassLoader也会委托给自己的父亲,于是BootstrapClassLoader就要去加载了,BootstrapClassLoader也想委托给自己的父类,可以它没有父类,因此就由自己加载,此时 Bootstrap就会搜索自己负责的标准库目录的相关的类,如果找到就加载,如果没找到,就由子类加载器进行加载..  ExtensionClassLoader 真正搜索扩展库相关的目录,如果找到就加载,如果没找到就由子类加载器加载.  ApplicationClassLoader,加载器进行加载(由于当前没有子类,如果没有找到,就会抛出 类找不到 这样的异常)

GC 垃圾回收机制

垃圾指的是不再使用的内存,垃圾回收,就是把不用的内存帮我们自动释放掉了. 而GC 就是一种主流的垃圾回收机制,GC垃圾回收机制 主要是针对 堆 里面的空间进行释放的, GC 是以"对象" 为基本单位,进行回收

GC 实际工作过程

1.找到垃圾/判定垃圾

哪个对象是垃圾,哪个对象不是垃圾,哪个对象以后可能还要使用,哪个对象后面不用了,关键思路是:看这个对象,有没有别的引用指向它(java中,使用对象只能通过引用来使用,如果一个对象没有引用指向,那么它一定不被使用)

具体如何判断对象是否有引用指向

1.可达性分析(Java中的做法)

Java 中的对象,都是通过引用来指向并访问的,经常是,一个引用指向一个对象,这个对象里的成员,又指向别的对象,比如链表,二叉树
整个Java中所有的对象,通过链表/树结构,整体串起来,可达性分析 就是把所有这些对象被组织的结构称为树,从树根节点出发,所有能被访问到的对象,标记成 "可达",不能访问到的,就是"不可达"
因此,通过上述标记JVM就可以知道所有可达的对象,剩下的不可达对象,就视为垃圾 进行回收

2.引用计数

给每个对象分配一个计数器(整数),每有一个引用指向该对象,计数器就+1,每次该引用被销毁 计数器就-1,引用计数为0时,此时这个对象就可以认为是垃圾了

2.清理垃圾

1.标记清除

直接把被标记的垃圾清除掉,缺点: 被释放的空间是闲散,零散,不连续,而我们申请内存需要连续的内存空间

2.复制算法

把不是垃圾的对象,复制到另一半,然后把刚刚有垃圾的一半整个空间删除掉.解决了内存碎片的问题
缺点,空间利用率低,如果要是垃圾少,有效对象多,复制成本大

3.标记整理

类似顺序表删除中间元素,把是垃圾的元素用不是垃圾的元素给填掉(元素搬运),再释放空间

分代回收(复制算法+标记整理)

根据不同的场景,使用不同的算法
分代: 基于经验规律,根据生命周期的长短,分别使用不同的算法
给对象引入一个 年龄 的概念,单位是 熬过GC垃圾回收的轮次,把年龄小的对象使用复制算法删除(年龄小的对象中,可能是垃圾的比较多),把年龄大的对象使用标记整理删除(老年代对象可能是垃圾较少)

更多推荐

激光焊接汽车PP塑料配件透光率测试仪

随着汽车主机厂对车辆轻量化的需求越来越强烈,汽车零部件轻量化设计、制造也成为汽车零部件生产厂商的重要技术指标。零部件企业要实现产品的轻量化,在材料指定的情况下,要通过产品设计优化、产品壁厚减小和装配方式的优化来解决。使用PP材料的汽车部品由于PP材料缩水率高,薄壁化设计会带来后续开发过程产品表面缩水、顶杆痕、应力痕等缺

最新Java JDK 21:全面解析与新特性探讨

🌷🍁博主猫头虎带您GotoNewWorld.✨🍁🦄博客首页——猫头虎的博客🎐🐳《面试题大全专栏》文章图文并茂🦕生动形象🦖简单易学!欢迎大家来踩踩~🌺🌊《IDEA开发秘籍专栏》学会IDEA常用操作,工作效率翻倍~💐🌊《100天精通Golang(基础入门篇)》学会Golang语言,畅玩云原生,走遍大

ChatGPT与日本首相交流核废水事件-精准Prompt...

了解更多请点击:ChatGPT与日本首相交流核废水事件-精准Prompt...https://mp.weixin.qq.com/s?__biz=Mzg2NDY3NjY5NA==&mid=2247490070&idx=1&sn=ebdc608acd419bb3e71ca46acee04890&chksm=ce64e42f

时序数据库 IoTDB 发布端边云原生解决方案,有效优化工业互联网数据上传时效与资源消耗...

2023年9月8日,由中国通信学会、福建省工业和信息化厅主办的2023中国国际工业互联网创新发展大会在厦门举办。大会主论坛中,时序数据库IoTDB发表其自研建立的端边云原生解决方案,该方案可实现端侧设备、边缘服务器、数据中心数据的协同汇聚,达到数据实时上传、带宽成本控制、与多终端同步管理。01关于大会为贯彻落实国家两个

大模型为使用Prompt提供的指导和建议

当使用大型语言模型时,合适的Prompt对于获取理想的响应至关重要。以下是一些常见任务的Prompt示例,以供参考:1.自然语言生成:“请为我生成一篇关于气候变化影响的文章。”“写一封感谢信,内容表达对某位导师的感激之情。”“编写一首诗,描述夏天的美丽。”2.问答任务:“回答以下问题:什么是人工智能?”“解释量子力学的

Java 泛型

目录(generic泛型)引言-idea技巧(快捷键)泛型的理解和好处传统方式用泛型来解决前面的问题(快速入门)泛型的好处泛型基本语法泛型介绍示例代码泛型的声明泛型的实例化(什么时候给泛型指定一个具体的类型)-一般来说是创建一个对象的时候指定的示例代码泛型语法和使用泛型使用案例自定义泛型自定义泛型类自定义泛型接口自定义

Prompt-To-Prompt——仅通过文本进行图像编辑

文章目录1.摘要2.算法2.1Cross-attentionintext-conditionedDiffusionModels2.2ControllingtheCross-attentionWordSwapAddingaNewPhraseAttentionRe–weighting3.应用Text-OnlyLocaliz

【C++】动态内存管理 ④ ( 对象的动态创建和释放引申思考 | 基础数据类型 内存分析 | malloc 分配内存 delete 释放 | new 分配内存 free 释放内存 )

文章目录一、对象的动态创建和释放引申思考二、基础数据类型内存分析1、malloc分配内存delete释放内存2、new分配内存free释放内存一、对象的动态创建和释放引申思考malloc和free是C语言stdlib标准库中的函数,用于分配和回收堆内存;new和delete是C++语言中的操作符,用于分配和回收堆内存;

[C++从入门到精通] 9.inline、const、mutable、this和static

📢博客主页:https://loewen.blog.csdn.net📢欢迎点赞👍收藏⭐留言📝如有错误敬请指正!📢本文由丶布布原创,首发于CSDN,转载注明出处🙉📢现在的付出,都会是一种沉淀,只为让你成为更好的人✨文章预览:一.特殊的成员函数inline二.成员函数末尾的const三.mutable四.返回

【Graph Net学习】LINE实现Graph Embedding

一、简介LINE(Large-scaleInformationNetworkEmbedding,2015)是一种设计用于处理大规模信息网络的算法。它主要的目标是在给定的大规模信息网络中学习高质量的节点嵌入,并尽量保留网络中信息的丰富性。其具体的表现为在一个低维空间里以向量形式表示网络中的节点,以便后续的机器学习任务可以

Kafka实现保证一批消息顺序生产消费的方案

背景ApacheKafka是一种高吞吐量的分布式发布订阅消息系统,它可以处理消费者和生产者之间的所有实时数据。在Kafka中,消息是以topic为单位进行归类的,而每个topic又可以分为多个partition,以实现数据的高效存储和并发处理。然而,由于Kafka的设计特性,消息在消费时并不能保证顺序。为了解决这个问题

热文推荐