看到JVM就不得不说垃圾回收,说到垃圾回收(Garbage Collection,GC),顾名思义,垃圾回收就是释放垃圾占用的空间,在Java中,程序员不需要去关心内存动态分配和垃圾回收的问题,这一切都交给了JVM来处理。那么在Java中,什么样的对象会被认定为“垃圾”?下面我们进行详细解析。

垃圾回收

当程序创建对象、数组等引用类型实体时,系统都会在堆内存中为之分配一块内存区,对象就保存在这块内存区中,当这块内存不再被任何引用变量引用时,这块内存就变成垃圾,等待垃圾回收机制进行回收。垃圾回收机制具有如下特征。

1.垃圾回收机制只负责回收内存中的对象,不会回收任何物理资源(例如数据库连接、网络IO等资源)

2.程序无法精确控制垃圾回收的运行,垃圾回收会在合适的时候进行。

3.在垃圾回收机制回收任何对象之前,总会先调用它的finalize()方法,该方法可能使该对象重新复活(让一个引用变量重新引用该对象),从而导致垃圾回收机制取消回收。

如果垃圾得不到及时的回收,就会出现内存泄漏的情况,具体的内存泄漏请先看这篇文章内存泄漏详细分析,将有助于理解本文中的对象状态。

对象的三种状态

当一个对象在堆内存中运行时,根据它被引用变量所引用的状态,可以把它所处的状态分成如下三种:

1433503642-5847815e7388b_articlex.png

可达状态:当一个对象被创建后,若有一个以上的引用变量引用它,则这个对象在程序中处于可达状态。

可恢复状态:如果程序中某个对象不再有任何引用变量引用它,它就进入了可恢复状态。此时,系统的垃圾回收机制准备回收该对象所占用的内存,在回收该对象之前,系统会调用所有可恢复状态对象的finalize()方法进行资源清理。如果系统在调用finalize()方法时重新让一个引用变量引用该对象,则这个对象会再次变成可达状态;否则该对象将进入不可达状态。

不可达状态:当对象与所有引用变量的关联都被切断,且系统已经调用所有对象的finalize()方法后依然没有使该对象变成可达状态,那么这个对象将永久性地失去引用,最后变成不可达状态。只有当一个对象处于不可达状态时,系统才会真正回收该对象所占有的资源。

举个例子

public static void test() {

String a = new String("C++");

a = new String("Java");

}

当程序执行test方法的第一行代码时,代码定义了一个a变量,并让该变量指向"C++"字符串,该代码执行结束后"C++"字符串对象处于可达状态。

当程序执行test方法的第二行代码时,代码再次创建了"Java"字符串对象,并让a变量指向该对象。此时"C++"字符串对象处于可恢复状态,而"Java"字符串处于可达状态。

总结:

一个对象可以被一个方法的局部变量引用,也可以被其他类的类变量引用,或被其他对象的实例变量引用。当某个对象被其他类的类变量引用时,只有该类被销毁后,该对象才会进入可恢复状态;当某个对象被其他对象的实例变量引用时,只有当该对象被销毁后,该对象才会进入可恢复状态。

GC如何找到“垃圾”对象

引用计数法:在java中是通过引用来和对象进行关联的,那么很显然一个简单的办法就是通过引用计数来判断一个对象是否可以被回收。如果一个对象引用数为0,则说明该对象基本不太可能在其他地方被使用到,那么这个对象就成为可被回收的对象了。

这种方式的特点是实现简单,而且效率较高,但是它无法解决循环引用的问题,因此在Java中并没有采用这种方式(Python采用的是引用计数法)。看下面这段代码:

public class Main {

public static void main(String[] args) {

MyObject object1 = new MyObject();

MyObject object2 = new MyObject();

object1.object = object2;

object2.object = object1;

object1 = null;

object2 = null;

}

}

class MyObject{

public Object object = null;

}

过程分析:

main方法中new了两个MyObject对象,这两个对象各有一个引用,一个是object1,一个是object2,然后object1和object2又把各自的属性object指向了对方,这时候两个MyObject对象引用数都为2,程序最后,分别将object1,object2释放引用,这时候两个MyObject对象的引用数由2-1成为1。这就导致它们的引用计数都不为0,那么垃圾收集器就永远不会回收它们。

可达性分析法:为了解决循环引用的问题,在Java中采取了可达性分析法。该方法的基本思想是通过一系列的“GC Roots”对象作为起点进行搜索,如果在“GC Roots”和一个对象之间没有可达路径,则称该对象是不可达的。

比较常见的将对象判定为可回收对象的情况:

1.显示地将某个引用赋值为null或者将已经指向某个对象的引用指向新的对象,比如下面的代码:

Object obj = new Object();(第一个对象)

obj = null;

Object obj1 = new Object();(第二个对象)

Object obj2 = new Object();(第三个对象)

obj1 = obj2;

上述代码执行完成后,第一个和第二个对象由于没有obj指向,所以可以进行回收。

2.局部引用所指向的对象,比如下面这段代码:

void fun() {

.....

for(int i=0;i<10;i++) {

Object obj = new Object();

System.out.println(obj.getClass());

}

}

循环每执行完一次,生成的Object对象都会成为可回收的对象。

对象的软、弱和虚引用

上面我们描述的都是用“=”来指向堆中的对象,是属于强引用,在java中还有其他类型的引用,那么他们和垃圾回收的关系是怎样的呢?

强引用(StrongReference)

Java程序中最常见的引用方式。程序创建一个对象,并把这个对象赋给一个引用变量,程序通过该引用变量来操作实际的对象。当一个对象被一个或一个以上的引用变量所引用时,它处于可达状态,不可能被系统垃圾回收机制回收。

软引用(SoftReference)

通过SoftReference类来实现,当一个对象只有软引用时,它有可能被垃圾回收机制回收。当系统内存空间足够时,它不会被系统回收,程序也可使用该对象;当系统内存空间不足时,系统可能会回收它。软引用通常用于对内存敏感的程序中。

弱引用(WeakReference)

通过WeakReference类实现,弱引用和软引用很像,但弱引用的引用级别更低。对于只有弱引用的对象而已,当系统垃圾回收机制运行时,不管系统内存是否足够,总会回收该对象所占用的内存。当然,并不是说当一个对象只有弱引用时,它就会立即被回收——正如那些失去引用的对象一样,必须等到系统垃圾回收机制运行时才会被回收。

虚引用(PhantomReference)

通过PhantomReference类实现,虚引用完全类似于没有引用。虚引用对对象本身没有太大影响,对象甚至感觉不到虚引用的存在。如果一个对象只有一个虚引用时,那么它和没有引用的效果大致相同。虚引用主要用于跟踪对象被垃圾回收的状态,虚引用不能单独使用,虚引用必须和引用队列(ReferenceQueue)联合使用。程序可以通过检查与虚引用关联的引用队列中是否已经包含了该虚引用,从而了解虚引用所引用的对象被系统垃圾回收过程。

java对象gc恢复_JVM垃圾回收与对象状态相关推荐

  1. java 2分代复制垃圾回收_Java对象的后事处理——垃圾回收(二)

    1 先谈Finalize() finalize()能做的所有工作,使用try-finally或者其他方式都可以做得更好.更及时,所以笔者建议大家完全可以忘掉Java语言中有这个方法的存在. --< ...

  2. 53.垃圾回收算法的实现原理、启动Java垃圾回收、Java垃圾回收过程、垃圾回收中实例的终结、对象什么时候符合垃圾回收的条件、GC Scope 示例程序、GC OutOfMemoryError的示例

    53.垃圾回收算法的实现原理 53.1.目录 53.2.启动Java垃圾回收 53.3.Java垃圾回收过程 53.4.垃圾回收中实例的终结 53.5.对象什么时候符合垃圾回收的条件? 53.5.1. ...

  3. 【趣话编程】一个Java对象的回忆录:垃圾回收

    简介:趣话编程第三期,今天让我们一起去看看一个Java对象的回忆录:垃圾回收. 原文链接 对象的诞生 "你醒啦!",迷迷糊糊中听到一个声音,我睁开了眼睛,发现一个小伙伴正看着我. ...

  4. Java基础知识强化83:System类之gc()方法(垃圾回收)以及和finalize()区别

    1. System概述: System类包含一些有用的类字段和方法.它不能被实例化. 2. gc()方法:垃圾回收器 1 public static void gc()       调用gc方法暗示着 ...

  5. Java中的垃圾回收与对象生命周期

    转载自   Java中的垃圾回收与对象生命周期 1. 垃圾回收 垃圾回收是Java程序设计中内存管理的核心概念,JVM的内存管理机制被称为垃圾回收机制. 一个对象创建后被放置在JVM的堆内存中,当永远 ...

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

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

  7. Java进阶 JVM 内存与垃圾回收篇(一)

    JVM 1. 引言 1.1 什么是JVM? 定义 Java Vritual Machine - java 程序的运行环境(Java二进制字节码的运行环境) 好处 一次编译 ,到处运行 自动内存管理,垃 ...

  8. JVM(3)之垃圾回收(GC垃圾收集器+垃圾回收算法+安全点+记忆集与卡表+并发可达性分析......)

    <深入理解java虚拟机>+宋红康老师+阳哥大厂面试题2总结整理 一.堆的结构组成 堆位于运行时数据区中是线程共享的.一个进程对应一个jvm实例.一个jvm实例对应一个运行时数据区.一个运 ...

  9. JAVA之JVM分代垃圾回收策略(一)

    一.为什么要分代 分代的垃圾回收策略,是基于这样一个事实:不同的对象的生命周期是不一样的.因此,不同生命周期的对象可以采取不同的收集方式,以便提高回收效率. 在Java程序运行的过程中,会产生大量的对 ...

最新文章

  1. maven 打包_一次打包引发的思考,原来maven还可以这么玩~
  2. Delphi 与 DirectX 之 DelphiX(95): TDIB.x
  3. twitter api 无法连接_光大银行牟健君:金融API的安全问题和应对技术
  4. 论文浅尝 | 基于知识图的问答变分推理
  5. 什么情况导致全表扫描,而不是用索引 收藏
  6. java洗扑克牌算法分析_IT兄弟连 Java语法教程 综合案例
  7. SQLMETAL :Linq对象生成
  8. 进击的UI------------UIToolBar(bottom导航条)
  9. Service层抽象规范
  10. c语言判断二叉树是不是二叉排序树_判断
  11. jenkins集成sonar问题记录
  12. oracle和mysql建表语句的区别_mysql和oracle建表语句的区别
  13. 如何提高关键词的质量度?
  14. 软件的接口设计图_面向服务架构(SOA)的汽车软件分析和设计
  15. 开源软件 AnimeGANv2:一键生成宫崎骏动漫风格照
  16. 键盘无法输入字母和数字,无法输入任何东西,但是键盘未损坏
  17. Prometheus(二)基础概念
  18. pip命令大全 含换源方法
  19. 第28章 Spring框架内的JNDI支持
  20. COBIT5给企业带来什么样的价值

热门文章

  1. php 获取子类的方法名,php获取分类下的所有子类方法
  2. u盘efi安装linux6.5,CentOS6.5安装的UEFI-GPT回退为MBR引导详解
  3. ListView gridivew.setOnItemClickListener无效
  4. 基于JAVA+SpringMVC+Mybatis+MYSQL的药店门店管理系统
  5. 编程迷宫_跟我学编程第十期——迷宫游戏
  6. 做三维模型_这几款倾斜实景三维裸眼3D采集软件你了解吗?
  7. floor mod sqlserver_ORACLE和SQLServer-SQL语句的区别
  8. .NET、C#和ASP.NET三者之间的区别与联系
  9. RobotFramework安装扩展库包autoitlibrary(四)
  10. Kubernetes v1.10.x HA 全手动安装教程(TL;DR)