Java 中的 "弱" 引用是什么?

作者:telami
来源:www.telami.cn/2017/weak-reference/

Java里一个对象obj被创建时,被放在堆里。当GC运行的时候,发现没有任何引用指向obj,那么就会回收obj对象的堆内存空间。

换句话说,一个对象被回收, 必须满足两个条件:

(1)没有任何引用指向它

(2)GC被运行。

在实际开发中,我们可以通过把所有指向某个对象的referece置空来保证这个对象在下次GC运行的时候被回收,类似下面:

Object c = new Car();    c=null;

但是,这样做是一件很繁琐并且违背GC自动回收原则的事。对于简单的情况, 手动置空是不需要程序员来做的, 因为在java中, 对于简单对象, 当调用它的方法执行完毕后, 指向它的引用会被从栈中弹出, 所以它就能在下一次GC执行时被回收了。

但是, 也有特殊例外. 当使用cache的时候, 由于cache的对象正是程序运行需要的, 那么只要程序正在运行, cache中的引用就不会被GC(或者说, cache中的reference拥有了和主程序一样的life cycle). 那么随着cache中的reference越来越多, GC无法回收的object也越来越多, 无法被自动回收。当这些object需要被回收时, 回收这些object的任务只有交给程序编写者了。然而这却违背了GC的本质(自动回收可以回收的objects)。

所以, java中引入了weak reference。

Object c = new Car(); //只要c还指向car object, car object就不会被回收 -->(强引用)

当一个对象仅仅被weak reference指向, 而没有任何其他strong reference指向的时候, 如果GC运行, 那么这个对象就会被回收。

下面这个是网上的例子,首先定义一个实体类:

public class Car {    private double     price;    private String    color;    public Car(double price, String color)    {        this.price = price;        this.color = color;    }    public double getPrice()    {        return price;    }    public String getColor()    {        return color;    }    public String toString()    {        return "This car is a " + this.color + " car, costs $" + price;    }}

一般使用WeakReference的时候都会定义一个类继承自WeakReference,在这个类中再定义一些别的属性,这里就不定义别的属性了:

public class WeakReferenceCar extends WeakReference<Car>{    public WeakReferenceCar(Car car)    {        super(car);    }}

main函数调用一下,当然为了更清楚地看到GC的效果,设置虚拟机参数”-XX:+PrintGCDetails”:

public static void main(String[] args){    Car car = new Car(2000.0, "red");    WeakReferenceCar wrc = new WeakReferenceCar(car);    wrc.setStr("111");    int i = 0;    while (true)    {        if (wrc.get() != null)        {            i++;            System.out.println("WeakReferenceCar's Car is alive for " + i + ", loop - " + wrc);        }        else        {            System.out.println("WeakReferenceCar's Car has bean collected");            break;        }    }}

最后是运行结果

WeakReferenceCar's Car is alive for 68450, loop - interview.WeakReferenceCar@776ec8dfWeakReferenceCar's Car is alive for 68451, loop - interview.WeakReferenceCar@776ec8dfWeakReferenceCar's Car is alive for 68452, loop - interview.WeakReferenceCar@776ec8dfWeakReferenceCar's Car is alive for 68453, loop - interview.WeakReferenceCar@776ec8df[GC (Allocation Failure) [PSYoungGen: 34304K->1000K(38400K)] 34320K->1016K(125952K), 0.0015129 secs] [Times: user=0.02 sys=0.02, real=0.00 secs] WeakReferenceCar's Car is alive for 68454, loop - interview.WeakReferenceCar@776ec8dfWeakReferenceCar's Car has bean collectedHeapPSYoungGen      total 38400K, used 1986K [0x00000000d5e00000, 0x00000000da900000, 0x0000000100000000)eden space 33280K, 2% used [0x00000000d5e00000,0x00000000d5ef6b70,0x00000000d7e80000)from space 5120K, 19% used [0x00000000d7e80000,0x00000000d7f7a020,0x00000000d8380000)to   space 5120K, 0% used [0x00000000da400000,0x00000000da400000,0x00000000da900000)ParOldGen       total 87552K, used 16K [0x0000000081a00000, 0x0000000086f80000, 0x00000000d5e00000)object space 87552K, 0% used [0x0000000081a00000,0x0000000081a04000,0x0000000086f80000)Metaspace       used 3547K, capacity 4564K, committed 4864K, reserved 1056768Kclass space    used 381K, capacity 388K, committed 512K, reserved 1048576K

可以看到在68454循环之后,WeakReferenceCar关联的对象Car被回收掉了,注意是弱引用关联的对象car被回收,而不是弱引用本身wrc被回收。

WeakReference的一个特点是它何时被回收是不可确定的, 因为这是由GC运行的不确定性所确定的. 所以, 一般用weak reference引用的对象是有价值被cache, 而且很容易被重新被构建, 且很消耗内存的对象.

在weak reference指向的对象被回收后, weak reference本身其实也就没有用了. java提供了一个ReferenceQueue来保存这些所指向的对象已经被回收的reference. 用法是在定义WeakReference的时候将一个ReferenceQueue的对象作为参数传入构造函数.

(0)

相关推荐

  • JVM真香系列:如何判断对象是否可被回收?

    回复"000"获取大量电子书 在JVM中程序寄存器.Java虚拟机栈.本地方法栈,这三个区是随着线程的创建而创建,随着线程结束而销毁. 其实就是这三个的生命周期和线程的生命周期一样 ...

  • java中引用数据类型有哪些?

    Java中有俩种数据类型,其中主要有8中基本数据类型和引用数据类型,除了8中基本数据类型以外都是引用数据类型,8中基本数据类型分别是byte,short,int,long,char,boolean,f ...

  • Java 8 中的方法引用,轻松减少代码量,提升可读性!

    Java 8 中的方法引用,轻松减少代码量,提升可读性!

  • Java中的引用到底是个什么玩意?

    文章目录 前言:先理解变量 变量有两类关键 变量的存储位置 再谈引用 基本数据类型与引用类型的区别 对引用的比喻 前言:先理解变量 在我们弄清楚引用之前,我们还得从变量讲起: 变量有两类关键 (1)变 ...

  • Java 中的关键字

    Java 中有多少个关键字,有大小写之分吗? Java 中有 48 个关键字在使用 + 两个保留关键字未使用,共 50 个关键字. Java 关键字全部都由是小写组成. Java 中保留关键字分别是哪 ...

  • 对象赋值在PHP中到底是不是引用?

    之前的文章中,我们说过变量赋值的问题,其中有一个问题是对象在进行变量赋值的时候,直接就是引用赋值.那么到底真实情况是怎样呢? 之前变量赋值的文章 PHP的变量赋值 对象引用测试 在继续深入的学习PHP ...

  • FastJson 处理json数据中对象相互引用,最后转为json字符串出现占位符("$ref"标识循环引用)"的问题

    环境 fastjson 1.2.41 问题说明 FastJson 问题 在json对象中有多个地方引用了相同的对象,在经过几次转换转为json字符串的时候会出现占位符, 然后使用fastjson 解析 ...

  • Java中的匿名内部类

    一.匿名内部类 之前的所有类都有自己的名字,但是有时候如果某个接口的实现类(或者某个父类的子类)只需要使用一次,此时这样的类如果我们单独定义出来则需要单独为其创建一个"*.java" ...

  • Java中的方法内部类

    一:方法内部类 就是在方法中直接定义一个内部类,之后直接使用这个内部类对象的方法,你作为语法 了解就行了. DEMO:方法中定义内部类     以上是在方法中定义了一个内部类,方法中的内部类能访问方法 ...

  • Java | 在 Java 中执行动态表达式语句: 前中后缀、Ognl、SpEL、Groovy、Jexl3

    在一些规则集或者工作流项目中,经常会遇到动态解析表达式并执行得出结果的功能. 规则引擎是一种嵌入在应用程序中的组件,它可以将业务规则从业务代码中剥离出来,使用预先定义好的语义规范来实现这些剥离出来的业 ...