【Java 虚拟机原理】垃圾回收算法 ( Java 虚拟机内存分区 | 垃圾回收机制 | 引用计数器算法 | 引用计数循环引用弊端 )

简介: 【Java 虚拟机原理】垃圾回收算法 ( Java 虚拟机内存分区 | 垃圾回收机制 | 引用计数器算法 | 引用计数循环引用弊端 )

文章目录

一、Java 虚拟机内存分区

二、垃圾回收机制

三、引用计数器算法 ( 无法解决循环引用问题 )





一、Java 虚拟机内存分区


Java 虚拟机内存分区 :


所有线程共有的内存区域 : 堆 , 方法区 ( 元空间 ) ;


方法区 : 存放 静态变量 , 常量 , class 编译后的代码数据 , 永久代对象数据 ( 很少 GC , 永久代的 GC 就是卸载常量池及类型 ) ;

堆 : 存放 创建的 对象实例 , 所有线程共享这个区域 , 堆是 垃圾回收的重点管理区域 ; 有 年轻代 , 老年代 空间 ;

年轻代 : 占整个 堆内存的 1 3 \cfrac{1}{3}

3

1


 的空间 ;

Eden 区 : 占 年轻代的 80 % 80\%80% 空间 ;

Survivor 区 : 占年轻代的 20 % 20\%20% 空间 ;

From : 占 年轻代的 10 % 10\%10% 空间 ;

To : 占 年轻代的 10 % 10\%10% 空间 ;

老年代 : 占整个 堆内存的 2 3 \cfrac{2}{3}

3

2


 空间 ; 老年代又称为永生代 , 只要程序没有 OOM 崩溃 , 这些 对象都是永生的 ; 比较大的对象直接放入老年代 ;

线程私有的内存区域 : 线程栈 , 本地方法栈 , 程序计数器 ; 每个线程都会分配上述 3 个内存区域 ;


虚拟机栈 ( 线程栈 ) : 主要管理 " 栈帧 " , 栈帧 在 线程栈 中也是 后入先出 的 ; 每个栈帧都包含 局部变量表 , 操作数栈 , 动态链接 , 方法出口 , 这四个数据 ;

动态链接 : 每个 " 栈帧 " 都包含一个引用 , 是该引用指向 运行时 , 该 栈帧 对应方法 的引用 ;

局部变量表 : 存放 方法中的 局部变量 ;

操作数栈 : 存放方法执行中的 操作数 , 一般是 0 , 1 , 2 0,1 , 20,1,2 个数值 , 很少有其它情况 ;

方法出口 : 记录本方法执行完毕后该执行的下一行 JVM 指令 ;

本地方法栈 : 记录 JNI 执行的栈 ;

程序计数器 : 记录线程执行到哪一行的 JVM 指令 ;





二、垃圾回收机制


Java 中 不需要 手动回收 对象内存 , 由 Java 虚拟机 自动执行 ;


好处 : 开发相对简单 ;

坏处 : 不能直接操作内存 ;

C 语言中 , 自己 malloc 手动申请了内存 , 必须使用 free 手动释放内存 ;



垃圾回收 的目标 是 垃圾对象 , 主要针对 堆内存 中的空间进行 ;


垃圾对象 的判断标准 : 没有任何作用的 被废弃的 对象 , 即 一个对象 没有任何引用 指向它 ;






三、引用计数器算法 ( 无法解决循环引用问题 )


引用计数器算法 判定 垃圾对象 : 使用引用计数算法 , 判定一个对象是否是垃圾对象 ;


为 对象 分配 引用计数器 , 当 有一个引用指向它时 , 引用计数器 + 1 + 1+1 , 如果有 3 33 个引用指向该对象时 , 引用计数器值为 3 33 , 如果某个引用失效 , 则 引用计数器 − 1 - 1−1 , 如果对象的全部引用失效 , 则对象的引用计数器值为 0 00 , 此时该对象就是 垃圾对象 , 需要被回收 ;


目前成熟的 Java 虚拟机 没有使用 引用计数器 算法 , 因为 如果 两个对象 仅仅是 互相引用 , 这两个对象的引用计数器 都是 1 11 , 但是这两个对象与其余对象没有任何联系 , 这样就导致了 这两个对象所占用的内存 , 永久泄漏了 ;


引用计数器算法 无解解决 循环引用的问题 , 如果有 2 22 个及以上的对象循环引用 , 与程序隔绝 , 造成内存泄漏 , 无法解决 ;


目录
相关文章
|
5天前
|
JavaScript 前端开发 Java
垃圾回收机制会导致内存泄漏吗?
【10月更文挑战第29天】虽然JavaScript的垃圾回收机制本身是为了有效地管理内存,但开发者在编写代码时需要注意上述这些可能导致内存泄漏的情况,遵循良好的编程习惯,及时释放不再使用的资源,以确保程序能够高效地利用内存资源,避免出现内存泄漏问题。
|
1天前
|
存储 算法 Java
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
本文详解自旋锁的概念、优缺点、使用场景及Java实现。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:什么是自旋锁?Java 实现自旋锁的原理?
|
1天前
|
Java
Java之CountDownLatch原理浅析
本文介绍了Java并发工具类`CountDownLatch`的使用方法、原理及其与`Thread.join()`的区别。`CountDownLatch`通过构造函数接收一个整数参数作为计数器,调用`countDown`方法减少计数,`await`方法会阻塞当前线程,直到计数为零。文章还详细解析了其内部机制,包括初始化、`countDown`和`await`方法的工作原理,并给出了一个游戏加载场景的示例代码。
Java之CountDownLatch原理浅析
|
3天前
|
Java 索引 容器
Java ArrayList扩容的原理
Java 的 `ArrayList` 是基于数组实现的动态集合。初始时,`ArrayList` 底层创建一个空数组 `elementData`,并设置 `size` 为 0。当首次添加元素时,会调用 `grow` 方法将数组扩容至默认容量 10。之后每次添加元素时,如果当前数组已满,则会再次调用 `grow` 方法进行扩容。扩容规则为:首次扩容至 10,后续扩容至原数组长度的 1.5 倍或根据实际需求扩容。例如,当需要一次性添加 100 个元素时,会直接扩容至 110 而不是 15。
Java ArrayList扩容的原理
|
1天前
|
缓存 算法 Java
本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制
在现代软件开发中,性能优化至关重要。本文聚焦于Java内存管理与调优,介绍Java内存模型、内存泄漏检测与预防、高效字符串拼接、数据结构优化及垃圾回收机制。通过调整垃圾回收器参数、优化堆大小与布局、使用对象池和缓存技术,开发者可显著提升应用性能和稳定性。
15 6
|
9天前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
24 2
|
9天前
|
算法 Java 数据库连接
Java连接池技术,从基础概念出发,解析了连接池的工作原理及其重要性
本文详细介绍了Java连接池技术,从基础概念出发,解析了连接池的工作原理及其重要性。连接池通过复用数据库连接,显著提升了应用的性能和稳定性。文章还展示了使用HikariCP连接池的示例代码,帮助读者更好地理解和应用这一技术。
24 1
|
3月前
|
存储 编译器 C语言
【C语言篇】数据在内存中的存储(超详细)
浮点数就采⽤下⾯的规则表⽰,即指数E的真实值加上127(或1023),再将有效数字M去掉整数部分的1。
337 0
|
15天前
|
存储 C语言
数据在内存中的存储方式
本文介绍了计算机中整数和浮点数的存储方式,包括整数的原码、反码、补码,以及浮点数的IEEE754标准存储格式。同时,探讨了大小端字节序的概念及其判断方法,通过实例代码展示了这些概念的实际应用。
29 1
|
19天前
|
存储
共用体在内存中如何存储数据
共用体(Union)在内存中为所有成员分配同一段内存空间,大小等于最大成员所需的空间。这意味着所有成员共享同一块内存,但同一时间只能存储其中一个成员的数据,无法同时保存多个成员的值。
下一篇
无影云桌面