一、Java中的强、软、弱、虚引用

在JDK中我们能够看到有一个java.lang.ref的包。这个包中就是Java中实现强、软、弱、虚引用的包,例如以下:

PhantomReference

虚引用:假设一个对象持有虚引用,就和没有持有引用一样,在不论什么时候都可能被垃圾回收器回收。

虚引用主要用来跟踪对象被垃圾回收的活动,虚引用另一个和弱、软引用不同的地方是虚引用必须和引用队列联合使用。当垃圾回收器准备回收一个对象时。假设发现它还有虚引用,就会在回收对象内存之前,把这个虚引用加入到与之关联的引用队列中。

程序能够通过推断引用队列中是否已经加入了虚引用,来知道对象是否将要被回收,这样我们就能够在对象被回收之前进行一些操作。

Reference

引用基类:这是一个抽象类。封装了关于引用的相关操作。如去除引用、比較、得到引用对象、推断引用地址是否相等。

ReferenceQueue

引用队列:在垃圾回收器回收对象之前,会将该引用加入到该队列。

SoftReference

软引用:假设内存空间不够用时,垃圾回收器就会回收该引用所引用到的内存对象。使用软引用能够实现内存敏感的快速缓存,软引用能够和引用队列联合使用。

WeakReference

弱引用:当垃圾回收器发现某个对象仅仅有弱引用时不管是否内存够用都会回收该对象,只是。因为垃圾回收器是一个优先级非常低的线程,因此不一定会非常快发现仅仅持有弱引用的对象。弱引用也能够和引用队列联合使用。

二、一个软(弱)引用的样例

以下我们来使用软引用读取一张图片,读出图片的宽和高,代码例如以下:

package 软引用;

import java.lang.ref.Reference;

/**
* 封装引用对象的基类
* @author CodeingSnail
*
* @param <T>
*/
public abstract class ReferenceObject<T>{ public Reference<T> ref; protected abstract T getInstance();
protected abstract Reference<T> getReference(T t); /*
* 获取引用对象及弱引用
*/
private T getRefrenceAndInstance(){
T t = getInstance();
getReference(t);
return t;
} /**
* 获取被引用的对象
* @return
*/
public T get(){
if(ref == null){
return getRefrenceAndInstance();
}
T t = ref.get();
if(t == null){
return getRefrenceAndInstance();
}
return t;
} /**
* 清空对象的引用。回收对象
*/
public void recycle() {
if(ref != null){
ref.clear();
ref = null;
}
}
}
import java.lang.ref.Reference;
import java.lang.ref.SoftReference; import javax.swing.ImageIcon; /**
* 引用图片资源的类
* @author Administrator
*
*/
public class ReferenceBitmap extends ReferenceObject<ImageIcon>{
String url;
public ReferenceBitmap(String url) {
this.url = url;
} @Override
protected Reference<ImageIcon> getReference(ImageIcon imageIcon) {
return new SoftReference<ImageIcon>(imageIcon);
} @Override
protected ImageIcon getInstance() {
return new ImageIcon(url);
} }
package 软引用;

public class Client {

	private ReferenceBitmap referenceBitmap;

	public static void main(String[] args) {
Client client = new Client();
client.referenceBitmap = new ReferenceBitmap("E:\\test.png");
System.out.println("图片的高度是:" + client.referenceBitmap.get().getIconHeight());
System.out.println("图片的宽度是:" + client.referenceBitmap.get().getIconWidth());
}
}

在上面样例中我们先定义了一个引用的抽象类,而且给子类提供两个回调方法用来创建图片对象和所须要的软引用(也能够是弱引用),在get()方法中获取对象的实例。事实上上面的样例是一个通用的方法。我们能够在getReference中定义我们须要的引用类型。假如我们如今已经非常明白,我们要的就是软引用,能够将代码简化例如以下:

public class Client {
SoftReference<ImageIcon> softRefrence;
private ReferenceBitmap referenceBitmap; public static void main(String[] args) {
Client client = new Client();
client.referenceBitmap = new ReferenceBitmap("E:\\test.png");
System.out.println("图片的高度是:" + client.referenceBitmap.get().getIconHeight());
System.out.println("图片的宽度是:" + client.referenceBitmap.get().getIconWidth()); //简化方法
client.softRefrence =
new SoftReference<ImageIcon>(new ImageIcon("E:\\test.png"));
System.out.println("图片的高度是:" + client.softRefrence.get().getIconHeight());
System.out.println("图片的宽度是:" + client.softRefrence.get().getIconWidth());
}
}

三、怎样配合引用队列使用

我们以下来分析一下java.util包下的WeakHashMap类,打开JDK后会发现对这个类有一个非常长的描写叙述。我们来一起看一下大概意思。
Hash table based implementation of the Map interface, with weak keys. An entry in a WeakHashMap will automatically be removed when its key is no longer in ordinary use. More precisely, the presence of a mapping for a given key will not prevent the key from being discarded by the garbage collector, that is, made finalizable, finalized, and then reclaimed. When a key has been discarded its entry is effectively removed from the map, so this class behaves somewhat differently from other Map implementations.
这句话的大概意思是WeakHashMap的哈希表是基于Map接口的。当中key中保存的是value的一个弱引用。当系统回收了该key所相应的实际对象之后。WeakHashMap会自己主动删除该key相应的key-value对。
    /**
* The entries in this hash table extend WeakReference, using its main ref
* field as the key.
*/
private static class Entry<K,V> extends WeakReference<Object> implements Map.Entry<K,V> {
V value;
final int hash;
Entry<K,V> next; /**
* Creates new entry.
*/
Entry(Object key, V value,
ReferenceQueue<Object> queue,
int hash, Entry<K,V> next) {
super(key, queue);
this.value = value;
this.hash = hash;
this.next = next;
} @SuppressWarnings("unchecked")
public K getKey() {
return (K) WeakHashMap.unmaskNull(get());
} public V getValue() {
return value;
} public V setValue(V newValue) {
V oldValue = value;
value = newValue;
return oldValue;
} public boolean equals(Object o) {
if (!(o instanceof Map.Entry))
return false;
Map.Entry<?,?> e = (Map.Entry<? ,?>)o;
K k1 = getKey();
Object k2 = e.getKey();
if (k1 == k2 || (k1 != null && k1.equals(k2))) {
V v1 = getValue();
Object v2 = e.getValue();
if (v1 == v2 || (v1 != null && v1.equals(v2)))
return true;
}
return false;
} public int hashCode() {
K k = getKey();
V v = getValue();
return Objects.hashCode(k) ^ Objects.hashCode(v);
} public String toString() {
return getKey() + "=" + getValue();
}
}

WeakHashMap.Entry继承自WeakReference,在构造方法中能够看到直接将key交给WeakReference并通过ReferenceQueue关联。

    /**
* Expunges stale entries from the table.
*/
private void expungeStaleEntries() {
for (Object x; (x = queue.poll()) != null; ) {
synchronized (queue) {
@SuppressWarnings("unchecked")
Entry<K,V> e = (Entry<K,V>) x;
int i = indexFor(e.hash, table.length); Entry<K,V> prev = table[i];
Entry<K,V> p = prev;
while (p != null) {
Entry<K,V> next = p.next;
if (p == e) {
if (prev == e)
table[i] = next;
else
prev.next = next;
// Must not null out e.next;
// stale entries may be in use by a HashIterator
e.value = null; // Help GC
size--;
break;
}
prev = p;
p = next;
}
}
}
}

WeakHashMap中有一个私有的expungeStaleEntries方法。会在大部分共同拥有方法中被调用,这种方法会将ReferenceQueue中全部失效的引用从Map中移除。WeakHashMap不会自己主动释放失效的引用,仅当包括了expungeStaleEntries方法被调用的时候才会释放。以下一个小样例来看一下WeakHashMap的使用:

	public static void main(String[] args) {
WeakHashMap<String, String> map = new WeakHashMap<String, String>();
map.put(new String("1"), "1");
map.put("2", "2");
String s = new String("3");
map.put(s, "3");
int i = 0;
while(map.size() > 0){
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("Map Size: " + map.size());
System.out.println(map.get("1"));
System.out.println(map.get("2"));
System.out.println(map.get("3"));
if(i == 3) s = null;
System.gc();
i++;
}
}

输出结果:

Map Size: 3
1
2
3
Map Size: 2
null
2
3
Map Size: 2
null
2
3
Map Size: 2
null
2
3
Map Size: 1
null
2
null
Map Size: 1
null

上面的样例中。第一个key外部没有强引用,则仅仅打印了一次就被回收器回收,第三个key有外部的强引用。当我们将外部引用去掉后也被垃圾回收器回收。第二个key是被字符串常量池所引用,所以一直存在。

Java中的软(弱)引用的更多相关文章

  1. java中强,软,弱,虚引用 以及WeakHahMap

    java中强,软,弱,虚引用  以及WeakHahMap   一:强软引用: 参考:http://zhangjunhd.blog.51cto.com/113473/53092/进行分析   packa ...

  2. JAVA中的四种引用以及ReferenceQueue和WeakHashMap的使用示例

    简介: 本文主要介绍JAVA中的四种引用: StrongReference(强引用).SoftReferenc(软引用).WeakReferenc(弱引用).PhantomReference(虚引用) ...

  3. Java中的四种引用

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

  4. JAVA中MESSAGEBOX,静态类直接引用

    原文:JAVA中MESSAGEBOX,静态类直接引用 package cisdi.mes.wrm.mcode.serviceImpl; import javax.persistence.Entity; ...

  5. Java中的基本类型和引用类型变量的区别

    Java中的基本类型和引用类型变量的区别   学了一年多,说实话你要我说这些东西我是真说不出来是啥意思     基本类型: 基本类型自然不用说了,它的值就是一个数字,一个字符或一个布尔值. 引用类型: ...

  6. java中的软引用,弱引用,虚引用

    http://zh.wikipedia.org/wiki/%E5%BC%B1%E5%BC%95%E7%94%A8 有些语言包含多种强度的弱引用.例如Java,在java.lang.ref[1]包中定义 ...

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

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

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

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

  9. java中的软,弱,虚引用介绍与特性分析

    java的弱,虚,软引用介绍 1.弱,虚,软引用的介绍 对于绝大部分的对象而言,在程序中是存在着一个引用变量引用该对象,这是常见的引用方式,也就是常说的 强引用,对于强引用引用的对象,系统JVM是不会 ...

随机推荐

  1. Mysql 5.6 慢日志配制

    一.配制my.cnf(/etc/my.cnf) slow_query_log=onlong_query_time=1slow_query_log_file=/var/log/slow_query.lo ...

  2. C#和C++中char类型的区别

    对于char,这个字符类型.我们一般都认为就是一个字节.今天在仔细比较发现,C#的char和C++的char是有区别的. 1.首先来看C#中char占多大空间 using System;using S ...

  3. pytest文档28-重复执行用例(pytest-repeat)

    前言 平常在做功能测试的时候,经常会遇到某个模块不稳定,偶然会出现一些bug,对于这种问题我们会针对此用例反复执行多次,最终复现出问题来. 自动化运行用例时候,也会出现偶然的bug,可以针对单个用例, ...

  4. Spring加载Hibernate 映射的几种方式及区别

    LocalSessionFactoryBean有好几个属性用来查找hibernate映射文件: mappingResources.mappingLocations.mappingDirectoryLo ...

  5. List 集合转换为String

    开发中会用到把 List<string>  的内容拼接成以逗号分隔的字符串的形式,现总结如下: 方法一: public String listToString(List list, cha ...

  6. 文件操作篇 close creat dup dup2 fcntl flock fsync lseek mkstemp open read sync write

    文件操作篇 close creat dup dup2 fcntl flock fsync lseek mkstemp open read sync write close(关闭文件) 相关函数 ope ...

  7. latex用法疑难解析

    latex用法疑难解析 1.问题:如何生成ps(PostScript)文件? 回答: 方法有二 (1)用dvips这个工具,在WinEdt编辑器中专门有一个按钮: (2)如果使用windows系统的话 ...

  8. OTL使用指南

    1 OTL简介 OTL 是 Oracle, Odbcand DB2-CLI Template Library 的缩写,是一个C++编译中操控关系数据库的模板库,它目前几乎支持当前所有的各种主流数据库, ...

  9. [转]【Delphi】 Thread.Queue与Synchronize的区别

    前话:  其实大家要学会看源码, 我接下来要说的这些东东,与其等别人讲,还不如自己搞几个代码试一下,印象还深刻点 TThread.Queue和TThread.Synchronize的区别,效果上:二者 ...

  10. jquery json实现二级动态联动

    以下为代码!需要导入json架包 function getCity1(){ var unitid = document.getElementById('addformunitid').value; $ ...