专栏首页Java爬坑系列Java强引用、软引用、弱引用及虚引用深入探讨

Java强引用、软引用、弱引用及虚引用深入探讨

强引用、软引用、弱引用和虚引用深入探讨

为了更灵活的控制对象的生命周期,在JDK1.2之后,引用被划分为强引用、软引用、弱引用、虚引用四种类型,每种类型有不同的生命周期,它们不同的地方就在于垃圾回收器对待它们会使用不同的处理方式。

引用类型在日常开发中并不常关注,也很少注意到,因此很多人忽略了它们的存在,而事实上,引用类型在Java体系中扮演着十分重要的角色,要想对Java体系有一个更深层次的理解,了解和掌握这些引用的用法是十分必要的。

在正式开始前,我们先来上两道开胃菜。

为什么需要回收

每一个Java程序中的对象都会占用一定的计算机资源,最常见的,如:每个对象都会在堆空间上申请一定的内存空间。但是除了内存之外,对象还会占用其它资源,如文件句柄,端口,socket等等。当你创建一个对象的时候,必须保证它在销毁的时候会释放它占用的资源。否则程序将会在OOM中结束它的使命。

在Java中不需要程序员来管理内存的分配和释放,Java有自动进行内存管理的神器——垃圾回收器,垃圾回收器会自动回收那些不再使用的对象。

在Java中,不必像C或者C++那样显式去释放内存,不需要了解其中回收的细节,也不需要担心会将同一个对象释放两次而导致内存损坏。所有这些,垃圾回收器都自动帮你处理好了。你只需要保证那些不再被使用的对象的所有引用都已经被释放掉了,否则,你的程序就会像在C++中那样结束在内存泄漏中。

虽然垃圾回收器确实让Java中的内存管理比C、C++中的内存管理容易许多,但是你不能对于内存完全不关心。如果你不清楚JVM到底会在什么条件下才会对对象进行回收,那么就有可能会不小心在代码中留下内存泄漏的bug。

因此,关注对象的回收时机,理解JVM中垃圾收集的机制,可以提高对于这个问题的敏感度,也能在发生内存泄漏问题时更快的定位问题所在。想了解更多关于垃圾回收相关的细节,可以参考这篇文章

为什么需要引用类型

引用类型是与JVM密切合作的类型,有些引用类型甚至允许在其引用对象在程序中仍需要的时候被JVM释放。

那么,为什么需要这些引用类型呢?

在Java中,垃圾回收器线程一直在默默的努力工作着,但你却无法在代码中对其进行控制。无法要求垃圾回收器在精确的时间点对某些对象进行回收。

有了这些引用类型之后,可以一定程度上增加对垃圾回收的粒度把控,可以让垃圾回收器在更合适的时机回收掉那些可以被回收掉的对象,而并不仅仅是只回收不再使用的对象。

这些引用类型各有特点,各有各的适用场景,清楚的了解和掌握它们的用法可以帮助你写出更加健壮的代码。

说明

在JDK 1.2以前的版本中,如果一个对象没有被任何变量引用,那么程序就无法再使用这个对象。也就是说,只有当对象处于可达(reachable)状态时,程序才能使用它。只有在对象没有任何其他对象引用它时,垃圾回收器才会对它进行收集。对象只有被引用和没有被引用两种状态。这种方式无法描述一些“食之无味,弃之可惜”的对象。

而很多时候,我们希望存在这样一些对象:当内存空间足够时,可以将它们保存在内存中,不进行回收;当内存空间变得紧张时,允许JVM回收这些对象。大部分缓存都符合这样的场景。

从JDK 1.2版本开始,Java对引用的概念进行了扩充,对象的引用分成了4种级别,从而使程序开发者能更加灵活地控制对象的生命周期,更好的控制创建的对象何时被释放和回收。

这4种级别由高到低依次为:强引用软引用弱引用虚引用

实力翻车

欢迎来到大型翻车现场,接下来将实力演示一波因为强引用过多导致的翻车例子。

如果你需要在整个程序运行期间保存一些对象(因为它们的初始化很耗费时间和资源),你可能会使用静态集合对象来存储并且在代码中随处使用它们。

public static Map<K, V> storedObjs = new HashMap<>();

但是这样,就会阻止垃圾回收器对集合中的对象进行回收和销毁。从而可能导致OOM的发生。例如:

public class OOMTest {
    public static List<Integer> cachedObjs = new ArrayList<>();
 
    public static void main(String[] args) {
        for (int i = 0; i < 100_000_000; i++) {
            cachedObjs.add(i);
        }
    }
}

输出如下:

Exception in thread “main” java.lang.OutOfMemoryError: Java heap space

这样就符合预期的翻车了。但你也许会说,谁会这么无聊,创建这么多变量。

嗯,确实是的,但是别忘了,一个程序可能会运行很长时间,几个月,甚至几年(如果你的代码和公司足够健壮的话),如果期间不断的创建变量而不清理的话(像上面那样把HashMap当缓存使用),是有可能会导致这种情况发生的。

内容编排

接下来的文章将从以下几方面对这四种引用进行介绍:

注意 本系列文章都是以JDK1.8 版本的代码进行分析,不同版本中代码会略有差异。

如果只是想要对这些引用进行简单了解,那么看完简要介绍部分即可,如果想要有更深入的研究,可以继续查阅源码剖析部分。

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

我来说两句

0 条评论
登录 后参与评论

相关文章

  • 你不可不知的Java引用类型之——弱引用

    弱引用是使用WeakReference创建的引用,弱引用也是用来描述非必需对象的,它是比软引用更弱的引用类型。在发生GC时,只要发现弱引用,不管系统堆空间是否足...

    弗兰克的猫
  • Java 强引用

    不要被这个强字吓到,以为这个引用就很厉害,其实强引用就是程序中使用的一般引用类型。举个简单的栗子:

    弗兰克的猫
  • 你不可不知的Java引用类型之——虚引用

    当试图通过虚引用的get()方法取得强引用时,总是会返回null,并且,虚引用必须和引用队列一起使用。既然这么虚,那么它出现的意义何在??

    弗兰克的猫
  • 什么样的对象需要被 GC ?

    上一篇文章 JVM 基本介绍 我们了解了一些基本的 JVM 知识,本篇开始逐步学习垃圾回收,我们都知道既然叫垃圾回收,那回收的就应该是垃圾,可是我们怎么知道哪些...

    周三不加班
  • JVM学习记录-对象已死吗

    前言 先来回顾一下,在jvm运行时数据区,分为两部分,一个部分是线程共享区,主要包括堆和方法区,另一部是线程私有区分包括本地方法栈,虚拟机栈和程序计数器。在线程...

    纪莫
  • 垃圾回收器如何处理循环引用

    垃圾回收是一门编程语言中必不可少的一部分,不论是手动释放内存的C和C++,还是自动回收垃圾的Java和C#等语言。对于Java这样的语言,一般的开发者不强求关心...

    技术小黑屋
  • Java Review (十八、面向对象----对象与垃圾回收)

    Java 内存运行时区域中的程序计数器、虚拟机栈、本地方法栈随线程而生灭;因此这几个区域的内存分配和回收都具备确定性,不需要过多考虑回收的问题,因为方法结束或者...

    三分恶
  • Java 的强引用、弱引用、软引用、虚引用

            强引用是使用最普遍的引用。如果一个对象具有强引用,那垃圾回收器绝不会回收它。如下:

    爱撸猫的杰
  • Java对象引用四个级别(强、软、弱、虚)

    最近,高级Java技术栈微信群中,有一些猿友在讨论JVM中对象的周期问题,有谈到引用的级别,现在为大家做个总结吧,虽然大多数公司并没有意识或者用到这些引用,但了...

    Java技术栈
  • AS3 内存回收机制

           AS3相对于以前版本的功能增强了很多,在赋予它重任时,同时也要它付出代价:垃圾收集器不再支持自动为你收集垃圾。本文中,我为大家整理了一些资料。首先...

    py3study

扫码关注云+社区

领取腾讯云代金券