深入理解java虚拟机(二)-----垃圾回收
做一个java程序员很是幸福,不用管不用的对象如何被回收,但是我认为了解一下也不是坏事。
一、如何判断对象已经死亡?
在进行垃圾回收之前,第一件事肯定是判断对象是否已经死亡。
1、引用计数算法
给对象添加一个引用计数器,当程序中使用到这个对象的时候,计数器+1;如果引用失效,计数器-1,当计数器为0时,说明程序中不再使用这个对象,既可以回收。
问题:试想一下,当A对象中使用B对象,B对象中有方法使用A对象,完犊子,两个对象永远存在。
2、可达性分析算法(主流的商用程序语言的主流实现,都是基于此算法)
通过一系列GC Roots的对象作为起始点,当一个对象到GC Roots没有任何引用链相连,说明对象不可用,可以被回收。

那么GC Roots对象的选择非常重要,主要包括下面几种:
1、虚拟机栈中引用的对象
2、方法区中类静态属性引用的对象
3、方法区中常量引用的对象
4、本地方法栈中JNI(native方法)引用的对象
二、引用
在jdk1.2之前,引用是这么定义的:
如果reference类型的数据中存储的数值代表的是另外一块内存的起始地址,就称这块内存代表着一个引用。
在jdk1.2之后,对引用进行了扩展:
1、强引用
Student student = new Student(),这种就叫做强引用,这种只要引用存在,垃圾收集器永远不会回收对象。
2、软引用
有用但不是必需的对象,在系统将要发生内存溢出异常之前,把这些对象列进回收范围进行第二次回收。SoftReference表示软引用。
3、弱引用
非必需的对象,强度比软引用更弱一些,弱引用对象只能生存到下一次垃圾收集发生之前。WeakReference表示弱引用。
4、虚引用
无法通过虚引用取得一个对象实例,存在的唯一意义就是在这个对象被收集器回收时收到一个系统通知。PhantomReference表示虚引用。
三、垃圾回收算法
垃圾回收的算法有很多,逐一介绍。
1、标记-清除算法
首先通过上面介绍的GC Roots算法判断对象是否存活,然后将不存活的对象打上标记。如下:


缺点:
- 效率问题,标记和清除两个过程的效率都不算高
- 资源浪费问题:从图中可以看到,清除完之后的内存都是不连续的,产生了很多的内存碎片,这样以后内存分配其他大对象的时候,就无法分配了。
2、复制算法
复制算法将内存分为大小相等的两块,每次只使用一块,当一块内存快满的时候,就将还存活的对象复制到另外一块内存上,然后把已使用的内存一次性清理掉,这样会提高效率。缺点则是浪费内存了,一次只能使用一半。


大部分的商业虚拟机都是采用这种算法还回收新生代,在之前的博文中提到了新生代主要可以分为eden、from survivor、to survivor,
通常eden:survivor=8:1,因为大部分新生代中的对象,大概能有98%是朝生夕死的。
3、标记-整理算法
标记过程基于标记-清除算法,但是整理的时候则不一样,让所有存活的对象都向一端移动,然后直接清理掉端边界以外的内存。


4、分代收集
以上几种算法各有各的特点,当前主流的商业虚拟机对于不同的内存区域,例如新生代、老年代,不同区域的对象特点不同,所以使用不同的算法达到最佳的效果。
四、内存分配
1、对象优先在eden区分配,如果eden区内存不够,则触发一次minor gc。
minor gc:年轻代内存回收被称为minor gc。
major gc:清理老年代。
full gc:清理整个堆空间—包括年轻代和老年代。
2、大对象直接进入老年代
大对象:需要大量连续内存空间的java对象,例如很长的字符串以及数组对象。
3、长期存活的对象将进入老年代
如何判断对象是长期存活的,其实也简单,jvm给对象维护了年龄计数器。对象从eden区没有被回收掉,年龄+1,到了survivor区还是没被干掉,年龄+1,达到年龄的阈值后,放入到老年代。
深入理解java虚拟机(二)-----垃圾回收的更多相关文章
- 深入理解Java虚拟机之垃圾回收篇
垃圾回收简介 Java 会对内存进行自动分配与回收管理,使上层业务更加安全,方便地使用内存实现程序逻辑.在不同的 JVM 实现及不同的回收机制中,堆内存的划分方式是不一样的. 简要地介绍下垃圾 ...
- 深入理解java虚拟机---3垃圾回收机制GC
本文来源于翁舒航的博客,点击即可跳转原文观看!!!(被转载或者拷贝走的内容可能缺失图片.视频等原文的内容) 若网站将链接屏蔽,可直接拷贝原文链接到地址栏跳转观看,原文链接:https://www.cn ...
- 深入理解java虚拟机【垃圾回收算法】
Java虚拟机的内存区域中,程序计数器.虚拟机栈和本地方法栈三个区域是线程私有的,随线程生而生,随线程灭而灭:栈中的栈帧随着方法的进入和退出而进行入栈和出栈操作,每个栈帧中分配多少内存基本上是在类结构 ...
- 深入理解Java虚拟机(三)——垃圾回收策略
所谓垃圾收集器的作用就是回收内存空间中不需要了的内容,需要解决的问题是回收哪些数据,什么时候回收,怎么回收. Java虚拟机的内存分为五个部分:程序计数器.虚拟机栈.本地方法栈.堆和方法区. 其中程序 ...
- 【深入理解Java虚拟机】垃圾回收
引用计数算法 给对象加一个计数器,引用一次+1,引用时效就-1,当计数器=0时对象就不能再被使用: 实现简单,判定效率高:Java虚拟接没有使用,主要原因是很难解决对象之间循环引用问题: GC算法: ...
- Java虚拟机之垃圾回收详解一
Java虚拟机之垃圾回收详解一 Java技术和JVM(Java虚拟机) 一.Java技术概述: Java是一门编程语言,是一种计算平台,是SUN公司于1995年首次发布.它是Java程序的技术基础,这 ...
- 每日一问:讲讲 Java 虚拟机的垃圾回收
昨天我们用比较精简的文字讲了 Java 虚拟机结构,没看过的可以直接从这里查看: 每日一问:你了解 Java 虚拟机结构么? 今天我们必须来看看 Java 虚拟机的垃圾回收算法是怎样的.不过在开始之前 ...
- Java虚拟机之垃圾回收
简述 Java与那些较传统的语言比如C++有个很大不同就是垃圾回收策略了.前者通常是虚拟机自动帮我们做了,而后者就需要我们手动来完成. Java虚拟机帮我们完成了垃圾回收,是不是意味着我们就不用完全去 ...
- Java 虚拟机 - GC 垃圾回收机制分析
Java 垃圾回收(Garbage Collection,GC) Java支持内存动态分配.垃圾自动回收,而 C++ 不支持.我想这可能也是 为什么 Java 脱胎于 C++ 的一个原因吧. GC 的 ...
随机推荐
- (ZT)算法杂货铺——k均值聚类(K-means)
https://www.cnblogs.com/leoo2sk/category/273456.html 4.1.摘要 在前面的文章中,介绍了三种常见的分类算法.分类作为一种监督学习方法,要求必须事先 ...
- linux命令行下xlsx转换成pdf或csv的笔记
使用libreoffice(可以用yum直接安装,占用了4xxM磁盘空间...) 然后命令行执行: 转换成csv,支持utf-8中文编码: libreoffice --invisible --con ...
- Linux gzip命令
语法: gzip [-acdfhlLnNqrtvV][-S <压缩字尾字符串>][-<压缩效率>][--best/fast][文件...] 或 gzip [-acdfhlLnN ...
- mysql null 相关bug
select 1 = null 并不是预期的 False 而是 Null select 1 is null; 这样才会产生预期的 False select 1 not is null 这样写是 错误 ...
- Codeforces Round #442 (Div. 2) E Danil and a Part-time Job (dfs序加上一个线段树区间修改查询)
题意: 给出一个具有N个点的树,现在给出两种操作: 1.get x,表示询问以x作为根的子树中,1的个数. 2.pow x,表示将以x作为根的子树全部翻转(0变1,1变0). 思路:dfs序加上一个线 ...
- Mysql设置大小写敏感
1.linux下mysql安装完后是默认:区分表名的大小写,不区分列名的大小写: 2.用root帐号登录后,在/etc/my.cnf 中的[mysqld]后添加添加lower_case_table_n ...
- 链接错误:multiple definition of 'xxx' 问题解决及其原理
内容借鉴 于CSDN炸鸡叔 错因 截图: “multiple definition of 'head' ” “multiple definition of 'tail' ” 解决过程: 1.首先要 ...
- jq动画设置图片抽奖
(因为自己是前端刚入门的小白所以中间出现很多问题,不过这对于我来说就是一次经验的积累) 预想效果:点击"开始",图片循环旋转,不是同时的效果,而是有一定的时间差.点击"开 ...
- 大数据学习笔记4 - Hadoop的优化与发展(Hadoop 2.0)
前面介绍了Hadoop核心组件HDFS和MapReduce,Hadoop发展之初在架构设计和应用性能方面仍然存在不足,Hadoop的优化与发展一方面体现在两个核心组件的架构设计改进,一方面体现在Had ...
- Huawei BGP和OSPF双边界重分布(二)
网络拓扑: 本例主要配置和例一致,主要是在AR3260-AR1和AR3260-AR2的路由域的边界上,从AR3260-AR1上重分布进BGP 65001的路由的时候打tag 650011,在AR326 ...