简介

本文主要介绍JAVA中的四种引用: StrongReference(强引用)、SoftReferenc(软引用)、WeakReferenc(弱引用)、PhantomReference(虚引用)的作用。同时我们还将介绍ReferenceQueue和WeakHashMap的功能和使用示例。

欢迎探讨,如有错误敬请指正

如需转载,请注明出处 http://www.cnblogs.com/nullzx/


1. JAVA中的四种引用

四种引用中,软引用、若引用、虚引用都需要相关类来创建。创建的时候都需要传递一个对象,然后通过引用的get方法获取真正的对象。

1.1 StrongReference(强引用)

强引用就是我们一般在程序中引用一个对象的方式

Object obj = new Object();

obj就是一个强引用。垃圾回收器绝不会回收它,当内存空间不足,Java虚拟机宁愿抛出OutOfMemoryError错误,使程序异常终止,也不会靠回收具有强引用的对象来解决内存不足的问题。

1.2 SoftReference(软引用)

软引用的创建要借助于java.lang.ref包下的SoftReferenc类。当JVM进行垃圾回收时,只有在内存不足的时候JVM才会回收仅有软引用指向的对象所占的空间。

package javalearning;

import java.lang.ref.SoftReference;
/*
* 虚拟机参数配置
* -Xms256m
* -Xmx1024m
*/
public class SoftReferenceDemo {
public static void main(String[] args){ /*软引用对象中指向了一个长度为300000000个元素的整形数组*/
SoftReference<int[]> softReference =
new SoftReference<int[]>(new int[300000000]); /*主动调用一次gc,由于此时JVM的内存够用,此时softReference引用的对象未被回收*/
System.gc();
System.out.println(softReference.get()); /*消耗内存,会导致一次自动的gc,此时JVM的内存不够用
*就回收softReference对象中指向的数组对象*/
int[] strongReference = new int[100000000]; System.out.println(softReference.get());
}
}

我们应该注意到,上面的代码中名为softReference的引用指向了一个
SoftReference对象,这个指向还是一个强引用类型。而SoftReference对象中指向int类型数组的引用就是一个软引用类型了。

运行结果

[I@2a139a55
null

1.3 WeakReference(弱引用)

弱引用的创建要借助于java.lang.ref包下的WeakReferenc类。当JVM进行垃圾回收时,无论内存是否充足,都会回收仅被弱引用关联的对象。由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现那些被弱引用指向的对象。

package javalearning;

import java.lang.ref.WeakReference;

public class WeakReferenceDemo {
public static void main(String[] args){ /*若引用对象中指向了一个长度为1000个元素的整形数组*/
WeakReference<String[]> weakReference =
new WeakReference<String[]>(new String[1000]); /*未执行gc,目前仅被弱引用指向的对象还未被回收,所以结果不是null*/
System.out.println(weakReference.get()); /*执行一次gc,即使目前JVM的内存够用,但还是回收仅被弱引用指向的对象*/
System.gc();
System.out.println(weakReference.get());
}
}

同理,上面的代码中名为weakReference的引用指向了一个
WeakReference对象,这个指向还是一个强引用类型。而WeakReference对象中指向String类型数组的引用就是一个弱引用类型了。

运行结果

[Ljava.lang.String;@2a139a55
null

1.4 PlantomReference(虚引用)

如果一个对象仅持有虚引用,那么它就和没有任何引用一样,在任何时候都可能被垃圾回收。创建一个虚引用对象时必须还要传递一个引用队列(ReferenceQueue)。

2. ReferenceQueue(引用队列)简介

当gc(垃圾回收线程)准备回收一个对象时,如果发现它还仅有软引用(或弱引用,或虚引用)指向它,就会在回收该对象之前,把这个软引用(或弱引用,或虚引用)加入到与之关联的引用队列(ReferenceQueue)中。如果一个软引用(或弱引用,或虚引用)对象本身在引用队列中,就说明该引用对象所指向的对象被回收了。

当软引用(或弱引用,或虚引用)对象所指向的对象被回收了,那么这个引用对象本身就没有价值了,如果程序中存在大量的这类对象(注意,我们创建的软引用、弱引用、虚引用对象本身是个强引用,不会自动被gc回收),就会浪费内存。因此我们这就可以手动回收位于引用队列中的引用对象本身。

除了上面代码展示的创建引用对象的方式。软、弱、虚引用的创建还有另一种方式,即在创建引用的同时关联一个引用队列。

SoftReference(T referent, ReferenceQueue<? super T> q)

WeakReference(T referent, ReferenceQueue<? super T> q)

PhantomReference(T referent, ReferenceQueue<? super T> q)

下面的示例中我们利用ReferenceQueue回收SoftReference对象本身。

package javalearning;

import java.lang.ref.ReferenceQueue;
import java.lang.ref.SoftReference; public class ReferenceQueneDemo { @SuppressWarnings({ "rawtypes", "unchecked" })
public static void main(String[] args){
/*创建引用队列*/
ReferenceQueue<SoftReference<int[]>> rq =
new ReferenceQueue<SoftReference<int[]>>(); /*创建一个软引用数组,每一个对象都是软引用类型*/
SoftReference<int[]>[] srArr = new SoftReference[1000]; for(int i = 0; i < srArr.length; i++){
srArr[i] = new SoftReference(new int[300000], rq);
} /*(可能)在gc前保留下了三个强引用*/
int[] arr1 = srArr[30].get();
int[] arr2 = srArr[60].get();
int[] arr3 = srArr[90].get(); /*占用内存,会导致一次gc,使得只有软引用指向的对象被回收*/
int[] strongRef = new int[200000000]; Object x;
int n = 0;
while((x = rq.poll()) != null){
int idx = 0;
while(idx < srArr.length){
if(x == srArr[idx]){
System.out.println("free " + x);
srArr[idx] = null; /*手动释放内存*/
n++;
break;
}
idx++;
}
} /*当然最简单的方法是通过isEnqueued()判断一个软引用方法是否在
* 队列中,上面的方法只是举例
int n = 0;
for(int i = 0; i < srArr.length; i++){
if(srArr[i].isEnqueued()){
srArr[i] = null;
n++;
}
}
*/
System.out.println("recycle " + n + " SoftReference Object");
}
}

运行结果(省略部分结果)

……
……
……
free java.lang.ref.SoftReference@cc285f4
free java.lang.ref.SoftReference@55f3ddb1
free java.lang.ref.SoftReference@8bd1b6a
free java.lang.ref.SoftReference@18be83e4
free java.lang.ref.SoftReference@cb5822
recycle 997 SoftReference Object

从上面的例子中可以看出,我们回收SoftReference对象的效率并不高。原因是每从队列中取出一个SoftReference引用,就是我们必须和SoftReference[]数组中的每一个对象逐个比较。这样的查找方式显然不及HashMap,所以我们自然想到构建一个引用类型的HashMap来解决这个问题。而实际上JDK中已经提供了一个具有这样功能的类,即WeakHashMap。

3. WeakHashMap简介

WeakHahsMap 的实现原理简单来说就是HashMap里面的条目 Entry继承了 WeakReference,那么当 Entry 的 key 不再被使用(即,引用对象不可达)且被 GC 后,那么该 Entry 就会进入到 ReferenceQueue 中。当我们调用WeakHashMap 的get和put方法会有一个副作用,即清除无效key对应的Entry。这个过程就和上面的代码很类似了,首先会从引用队列中取出一个Entry对象,然后在HashMap中查找这个Entry对象的位置,最后把这个 Entry 从 HashMap中删除,这时key和value对象都被回收了。重复这个过程直到队列为空。

最后说明一点,WeakHashMap是线程安全的。

package javalearning;

import java.util.WeakHashMap;

public class WeakHashMapDemo {
public static void main(String[] args){ WeakHashMap<String, byte[]> whm = new WeakHashMap<String, byte[]>();
String s1 = new String("s1");
String s2 = new String("s2");
String s3 = new String("s3"); whm.put(s1, new byte[100]);
whm.put(s2, new byte[100]);
whm.put(s3, new byte[100]); s2 = null;
s3 = null; /*此时可能还未执行gc,所以可能还可以通过仅有弱引用的key找到value*/
System.out.println(whm.get("s1"));
System.out.println(whm.get("s2"));
System.out.println(whm.get("s3")); System.out.println("-------------------"); /*执行gc,导致仅有弱引用的key对应的entry(包括value)全部被回收*/
System.gc();
System.out.println(whm.get("s1"));
System.out.println(whm.get("s2"));
System.out.println(whm.get("s3"));
}
}

运行结果

[B@2a139a55
[B@15db9742
[B@6d06d69c
-------------------
[B@2a139a55
null
null

4. 参考内容

[1] Java中的强引用,软引用,弱引用,虚引用有什么用?

[2] ReferenceQueue的使用

[3] 强软弱虚---强引用、软引用、弱引用、虚引用

JAVA中的四种引用以及ReferenceQueue和WeakHashMap的使用示例的更多相关文章

  1. Java中的四种引用

    引用定义 实际上,Java中存在四种引用,它们由强到弱依次是:强引用.软引用.弱引用.虚引用.下面我们简单介绍下这四种引用: 强引用(Strong Reference):通常我们通过new来创建一个新 ...

  2. Java 中的四种引用及垃圾回收策略

    Java 中有四种引用:强引用.软引用.弱引用.虚引用: 其主要区别在于垃圾回收时是否进行回收: 1.强引用 使用最普遍的引用.如果一个对象具有强引用,那就 类似于必不可少的生活用品,垃圾回收器绝不会 ...

  3. Java中的四种引用方式

      无论是通过引用计数算法判断对象的引用数量,还是通过可达性分析算法判断对象的引用链是否可达,判定对象是否存活都与"引用"有关.在Java语言中,将引用又分为强引用.软引用.弱引用 ...

  4. Java 中的四种引用

    1.强引用(Strong Reference)在 Java 中四种引用中是“最强”的,我们平时通过 new 关键字创建的对象都属于强引用,如下面的代码: Person person = new Per ...

  5. Java入门系列 Java 中的四种引用

    Why java内存管理分为内存分配和内存回收,都不需要程序员负责,垃圾回收的机制主要是看对象是否有引用指向该对象. java对象的引用包括强引用,软引用,弱引用,虚引用 Java中提供这四种引用类型 ...

  6. java中的四种引用方式(强引用,软引用,弱引用,虚引用)

    java内存管理主要有内存分配和内存回收,都不需要程序员负责,垃圾回收的机制主要是看对象是否有引用指向该对象. java中对象的引用主要有四种:强引用,软引用,弱引用,虚引用. Java中提供这四种引 ...

  7. Java中的四种引用(强引用、软引用、弱引用、虚引用)

    以下内容摘自<深入理解Java虚拟机 JVM高级特性与最佳实践>第2版,强烈推荐没有看过的同学阅读,读完的感觉就是"原来学的都是些什么瘠薄东西(╯‵□′)╯︵┴─┴" ...

  8. Java中的四种引用和引用队列

    目录 强引用 软引用 弱引用 幻象引用 Reachability Fence 参考 强引用 正常的引用,生命周期最长,例如 Object obj = new Object(); 当JVM内存不足时,宁 ...

  9. JAVA基础学习之throws和throw的区别、Java中的四种权限、多线程的使用等(2)

    1.throws和throw的区别 throws使用在函数外,是编译时的异常,throw使用在函数内,是运行时的异常 使用方法 public int method(int[] arr) throws ...

随机推荐

  1. (转载)java基础:关于java流与文件操作

    原文摘自: http://www.blogjava.net/haizhige/archive/2008/08/03/219668.html 1.描述:流是字节数据或字符数据序列. Java采用输入流对 ...

  2. windows调试工具列表

    摘自windbg帮助文档(windbg中输入.hh): Debugging Tools for Windows (安装WinDbg后这些工具都会安装在目录C:\Program Files (x86)\ ...

  3. java字符串,包,数组及空心正方形,菱形的实例

    一.数组:相同类型的多个对像引用类型:所有的类,接口,数组,int[] ints(变量名) = new int[3]new:指的是在内存空间重新开辟一块区域 String s1 = "abc ...

  4. HDU 5547 Sudoku(DFS)

    题目网址:http://acm.hdu.edu.cn/showproblem.php?pid=5547 题目: Sudoku Time Limit: 3000/1000 MS (Java/Others ...

  5. 小哈学Python第三课-字符集编码

    table.hovertable { font-family: verdana, arial, sans-serif; font-size: 11px; color: #333333; border- ...

  6. ubuntu14.04_CUDA8.0_cudnn5.1_Tensorflow配置

    深度学习框架tensorflow相比与caffe抽象层做的更好,即使用tensorflow的人不需要关心底层的实现,做底层实现的人不需要关心上层的模型和算法;caffe耦合比较紧凑,若想caffe用的 ...

  7. Abp(.NetCore)开发与发布过程2

    在Abp(.NetCore)开发过程中遇到很多问题,今天记录下Abp的防CSRF功能(AntiForgeryToken ), 背景知识. AntiForgeryToken 可以说是处理/预防CSRF的 ...

  8. 前端性能优化--图片懒加载(lazyload image)

    话说前头: 上次写了一篇webpack的学习心得,webpack能做到提升前端的性能,其模块打包最终生成一个或少量的文件能够减少对服务端的请求.除此之外,本次的图片懒加载(当然不仅限于图片,还可以有视 ...

  9. cocos2dx之WebView踩过的坑(android返回键处理问题)

    最近游戏接入了一个私服平台,由于没有sdk,所以支付相关的操作需要在网页端进行,也就是说点击充值需要在游戏内部弹出一个网页,并定位到平台充值的地址.查阅相关资料后决定使用cocos2dx自带的WebV ...

  10. JS浏览器对象:window对象、History、Location对象、Screen对象

    一.JS浏览器对象-window 1.window对象 window对象是BOM的核心,window对象指当前的浏览器窗口 所有JavaScript全局对象.函数以及变量均自动成为window对象的成 ...