先说一些题外话,Java虚拟机在执行Java程序的过程中会把它所管理的内存划分为若干个不同的数据区,这些区分为线程私有区和线程共享区

1、线程私有区

a、程序计数器

记录正在执行的虚拟机字节码指令地址。此区域是是唯一一个在java虚拟机规范中没有规定任何OutOfMemoryError情况的区域。

b、Java虚拟机栈

描述的是Java方法执行的内存模型,每个方法在执行的同时会创建一个栈帧

c、本地方法栈

它与虚拟机栈发挥的作用是类似的,它们之间的区别不过是虚拟机栈为虚拟机执行java方法(也就是字节码)服务,而本地方法栈则为虚拟机使用的Native方法服务。

2、线程共享区

a、Java堆

被所有线程共享的一块内存区域,也是Java虚拟机所管理的内存中最大的一块。

b、方法区

用于存储已被虚拟机加载的类信息、常量、静态变量、即时编辑器编译后的代码等数据,虽然Java虚拟机规范把方法区描述为堆的一个逻辑部分,但是它却有一个别名Non-Heap(非堆)

下面开始说正题

目前虚拟机基本都是采用可达性算法,为什么不采用引用计数算法呢?下面就说说引用计数法是如何统计所有对象的引用计数的,再对比分析可达性算法是如何解决引用技术算法的不足。先简单说说这两个算法:

1、引用计数法(reference-counting):每个对象都有一个引用计数器,当对象被引用一次,计数器就加1,当对象引用时效一次就减,当计数器为0,意味着对象是垃圾对象,可以被GC回收。

2、可达性算法(GC Root Tracing):从GC Root作为起点开始搜索,那么整个连通图中对象都是活的,对于GC Root无法达到的对象便是垃圾对象,随时可被GC回收。

采用引用计数算法的系统只需在每个实例对象创建之初,通过计数器来记录所有的引用次数即可。而可达性算法,则需要再次GC时,遍历整个GC根节点来判断是否回收。

下面通过一段代码来对比说明:

public classGcDemo {public static voidmain(String[] args) {//分为6个步骤

GcObject obj1 = new GcObject(); //Step 1

GcObject obj2 = new GcObject(); //Step 2

obj1.instance= obj2; //Step 3

obj2.instance = obj1; //Step 4

obj1= null; //Step 5

obj2 = null; //Step 6

}

}classGcObject{public Object instance = null;

}

1、引用计数算法

如果采用的是引用计数算法:

再回到前面代码GcDemo的main方法共分为6个步骤:

Step1:GcObject实例1的引用计数加1,实例1的引用计数=1;

Step2:GcObject实例2的引用计数加1,实例2的引用计数=1;

Step3:GcObject实例2的引用计数再加1,实例2的引用计数=2;

Step4:GcObject实例1的引用计数再加1,实例1的引用计数=2;

执行到Step 4,则GcObject实例1和实例2的引用计数都等于2。

接下来继续结果图:

Step5:栈帧中obj1不再指向Java堆,GcObject实例1的引用计数减1,结果为1;

Step6:栈帧中obj2不再指向Java堆,GcObject实例2的引用计数减1,结果为1。

到此,发现GcObject实例1和实例2的计数引用都不为0,那么如果采用的引用计数算法的话,那么这两个实例所占的内存将得不到释放,这便产生了内存泄露。

2、可达性算法

这是目前主流的虚拟机都是采用GC Roots Tracing算法,比如Sun的Hotspot虚拟机便是采用该算法。 该算法的核心算法是从GC Roots对象作为起始点,利用数学中图论知识,图中可达对象便是存活对象,

而不可达对象则是需要回收的垃圾内存。这里涉及两个概念,一是GC Roots,一是可达性。

那么可以作为GC Roots的对象(见下图):

虚拟机栈的栈帧的局部变量表所引用的对象;

本地方法栈的JNI所引用的对象;

方法区的静态变量和常量所引用的对象;

关于可达性的对象,便是能与GC Roots构成连通图的对象,如下图:

从上图,reference1、reference2、reference3都是GC Roots,可以看出:

reference1-> 对象实例1;

reference2-> 对象实例2;

reference3-> 对象实例4;

reference3-> 对象实例4 -> 对象实例6;

可以得出对象实例1、2、4、6都具有GC Roots可达性,也就是存活对象,不能被GC回收的对象。

而对于对象实例3、5直接虽然连通,但并没有任何一个GC Roots与之相连,这便是GC Roots不可达的对象,这就是GC需要回收的垃圾对象。

到这里,相信大家应该能彻底明白引用计数算法和可达性算法的区别吧。

再回过头来看看最前面的实例,GcObject实例1和实例2虽然从引用计数虽然都不为0,但从可达性算法来看,都是GC Roots不可达的对象。

总之,对于对象之间循环引用的情况,引用计数算法,则GC无法回收这两个对象,而可达性算法则可以正确回收。

java可达性_java垃圾回收机制--可达性算法相关推荐

  1. java gc 可达性_JAVA--GC 垃圾回收机制----可达性分析算法

    在JVM 中,java 为我们提供可有效的垃圾回收机制,GC ,GC的创建无疑是为了缓解内存压力.保存有效数据.回收垃圾无效数据: 在此之前GC在我的理解中,一直只是个概念,内存中出现垃圾,GC来回收 ...

  2. 第八章软件构造的性能——构造性能的度量、原则与方法(java中的垃圾回收机制及算法)

    这节内容真的是多到炸裂,而且全都是概念,不过很挺有用的,学完这节会对内存管理有一个比较深的理解和认识,就是--这也太多了--嘤嘤嘤T_T 1.性能度量(performance metrics) 第一点 ...

  3. GC:垃圾回收机制及算法

    GC:垃圾回收机制及算法 关键词 算法:标记(清除/复制/整理).分代收集 收集器(Serial[串行].ParNew[并行].Parallel Scavenge[并行].Serial Old[串行] ...

  4. 【JVM】垃圾回收机制及算法

    垃圾回收机制及算法 一.垃圾回收概述 二.对象是否存活 1. 判断对象是否存活 - 引用计数算法 2.判断对象是否存活-可达性分析算法 1.可达性分析算法 2.JVM之判断对象是否存活 3.关于引用 ...

  5. 一篇文章搞定java中的垃圾回收机制面试题

    一篇文章搞定java中的垃圾回收机制面试题 任何语言在运行过程中都会创建对象,也就意味着需要在内存中为这些对象在内存中分配空间,在这些对象失去使用的意义的时候,需要释放掉这些内容,保证内存能够提供给新 ...

  6. java gc 有钱人_小猿圈java之GC垃圾回收机制

    垃圾回收机制是提高性能的重要机制,所以不管学什么语言也好,垃圾回收机制都是有的,也是很重要的,小猿圈这章详解一下java的GC垃圾回收机制,需要用的或者感兴趣的朋友们,可以跟着小编学习一下. 一.GC ...

  7. 6种java垃圾回收算法_Java垃圾回收机制

    Java垃圾回收机制 说到垃圾回收(Garbage Collection,GC),很多人就会自然而然地把它和Java联系起来.在Java中,程序员不需要去关心内存动态分配和垃圾回收的问题,这一切都交给 ...

  8. java垃圾回收机制标记_Java的垃圾回收机制-垃圾收集算法(一)

    当需要排查各种内存溢出,内存泄漏等问题时,当垃圾收集成为系统达到更高并发的瓶颈时,我们有必要深入GC的原理. image.png 常见垃圾回收算法 在查看垃圾回收具体过程的时候,运行程序加上: -XX ...

  9. java垃圾回收机制串行_Java垃圾回收机制

    Java语言是一门自动内存管理的语言,不再需要的对象可以通过垃圾回收自动进行内存释放. Java运行时内存区域划分 JVM将Java程序运行时内存区域划分成以下几个部分: 程序计数器(Program ...

最新文章

  1. 【C++】clipp 一个命令行参数解析器
  2. PFLD:简单、快速、超高精度人脸特征点检测算法
  3. 了解如何通过Python使用SQLite数据库
  4. mysql 触发器定义变量_MySQL 函数存储过程触发器定义简单示例
  5. Linux:建立内核代码树
  6. 学安全工程用不用计算机,上重点大学的末流专业,不如上普通大学的重点专业,你赞成吗?...
  7. 【英语学习】【Daily English】U10 Education L01 Is this certificate a must?
  8. POJ3617 Best Cow Line【水题】
  9. Error: Cannot find module ‘webpack-cli/bin/config-yargs‘ 解决办法
  10. Thinking in Java 14.3.3递归计数
  11. 值类型和引用类型 区别
  12. Spring开发指南0.8版(By夏昕)
  13. 智慧工地实名制劳务管理系统
  14. Appium+python自动化(二十一)- 让猴子按你指令大闹手机,让我们都成为耍猴高手(超详解)...
  15. 大彩科技串口屏串口通讯问题解决记录
  16. windows11中如何显示此电脑图标
  17. Android 开发多摄像头 API
  18. Abaqus中批量对节点施加集中力荷载
  19. 【区块链】区块链是什么?20问:读懂区块链
  20. android 修改 dpi_魅族16th|魔趣100|安卓10.0|归属地|机型修改|性能调整|稳定流畅

热门文章

  1. 配置SAMBA文件共享的基本方法
  2. java ++i、i++、i++ + ++i、i++ + i++原理
  3. 移动端导出excel_连载系列【4】Excel开发移动端quot;APPquot;
  4. 基于matlab的大米,大米颗数计算MATLAB软件
  5. matlab 归一化_机器学习中如何用Fscore进行特征选择(附Matlab代码)
  6. tornado学习笔记day02-进阶与提升
  7. 【Kubernetes】k8s 的基本使用指令
  8. Linux系统7个运行级别图文详解
  9. Log 日志的使用与重要性
  10. WIFI无线路由器的五种工作模式