Java中的软(弱)引用

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

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

Java中的软引、弱、虚引用

PhantomReference

虚引用:如果一个对象持有虚引用,就和没有持有引用一样,在任何时候都可能被垃圾回收器回收。虚引用主要用来跟踪对象被垃圾回收的活动,虚引用还有一个和弱、软引用不动的地方是虚引用必须和引用队列联合使用。当垃圾回收器准备回收一个对象时,如果发现它还有虚引用,就会在回收对象内存前把这个虚引用加入到与之关联的引用队列中。程序可以通过判断引用队列是否已经添加了虚引用,来知道对象是否将要被回收,这样我们就可以在对象被回收之前进行一些操作。

Reference

这个是引用的基类,是一个抽象类,封装了关于引用的相关操作,如:去除引用、比较、获取引用对象、判断引用地址是否相同。

ReferenceQueue

引用队列:在垃圾回收器回收对象之前,会将对应关联的引用添加到该队列。

SoftReference

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

WeakReference

弱引用:当垃圾回收器发现某个对象只有弱引用时无论是否内存够用都会回收该对象,不过,由于垃圾回收器是一个优先级很低的线程,因此不一定会很快发现只持有弱引用的对象,弱引用也可以和引用队列联合使用。

我们来总结一下:

引用类型被垃圾回收时间用途生存时间对应包装类
强引用从来不会对象的一般状态JVM停止运行时终止正常类
软引用当内存不足时对象缓存内存不足时终止SoftReference
弱引用正常垃圾回收时对象缓存垃圾回收后终止WeakReference
虚引用正常垃圾回收时跟踪对象的垃圾回收垃圾回收后终止PhantomReference

一个软(弱)引用的例子

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
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;
}
}
}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
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);
}

}
1
2
3
4
5
6
7
8
9
10
11
12
13
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中定义我们需要的引用类型。假如我们现在已经很明确,我们要的就是软引用,可以将代码简化如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
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对。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
/** 
* 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关联。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
/** 
* 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的使用:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
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++;
}
}

输出结果:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
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是被字符串常量池所引用,所以一直存在。

评论

Ajax Android AndroidStudio Animation Anroid Studio AppBarLayout Babel Banner Buffer Bulma ByteBuffer C++ C11 C89 C99 CDN CMYK COM1 COM2 CSS Camera Raw, 直方图 Chrome Class ContentProvider CoordinatorLayout C语言 DML DOM Dagger Dagger2 Darktable Demo Document DownloadManage ES2015 ESLint Element Error Exception Extensions File FileProvider Flow Fresco GCC Git GitHub GitLab Gradle Groovy HTML5 Handler HandlerThread Hexo Hybrid I/O IDEA IO ImageMagick IntelliJ Intellij Interpolator JCenter JNI JS Java JavaScript JsBridge Kotlin Lab Lambda Lifecycle Lint Linux Looper MQTT MVC MVP Maven MessageQueue Modbus Momentum MySQL NDK NIO NexT Next Nodejs ObjectAnimator Oracle VM Permission PhotoShop Physics Python RGB RS-232 RTU Remote-SSH Retrofit Runnable RxAndroid RxJava SE0 SSH Spring SpringBoot Statubar Style Task Theme Thread Tkinter UI UIKit UML VM virtualBox VS Code VUE ValueAnimator ViewPropertyAnimator Vue Vue.js Web Web前端 Workbench api apk bookmark by关键字 cli compileOnly computed css c语言 databases demo hexo hotfix html iOS icarus implementation init jQuery javascript launchModel logo merge methods mvp offset photos pug query rxjava2 scss servlet shell svg tkinter tomcat transition unicode utf-8 vector virtual box vscode watch webpack 七牛 下载 中介者模式 串口 临潼石榴 主题 书签 事件 享元模式 仓库 代理模式 位运算 依赖注入 修改,tables 光和色 内存 内核 内部分享 函数 函数式编程 分支 分析 创建 删除 动画 单例模式 压缩图片 发布 可空性 合并 同向性 后期 启动模式 命令 命令模式 响应式 响应式编程 图层 图床 图片压缩 图片处理 图片轮播 地球 域名 基础 增加 备忘录模式 外观模式 多线程 大爆炸 天气APP 太白山 头文件 奇点 字符串 字符集 存储引擎 宇宙 宏定义 实践 属性 属性动画 岐山擀面皮 岐山肉臊子 岐山香醋 工具 工厂模式 年终总结 开发技巧 异常 弱引用 恒星 打包 技巧 指令 指针 插件 插值 摄影 操作系统 攻略 故事 数据库 数据类型 数组 文件 新功能 旅行 旋转木马 时序图 时空 时间简史 曲线 杂谈 权限 枚举 架构 查询 标准库 标签选择器 样式 核心 框架 案例 桥接模式 检测工具 模块化 模板 模板引擎 模板方法模式 油泼辣子 泛型 洛川苹果 浅色状态栏 渲染 源码 源码分析 瀑布流 热修复 版本 版本控制 状态栏 状态模式 生活 留言板 相册 相对论 眉县猕猴桃 知识点 码云 磁盘 科学 笔记 策略模式 类图 系统,发行版, GNU 索引 组件 组合模式 绑定 结构 结构体 编码 网易云信 网格布局 网站广播 网站通知 网络 美化 联合 脚手架 膨胀的宇宙 自定义 自定义View 自定义插件 蒙版 虚拟 虚拟机 补码 补齐 表单 表达式 装饰模式 西安 观察者模式 规范 视图 视频 解耦器模式 设计 设计原则 设计模式 访问者模式 语法 责任链模式 贪吃蛇 转换 软件工程 软引用 运算符 迭代子模式 适配器模式 选择器 通信 通道 配置 链表 锐化 错误 键盘 闭包 降噪 陕西地方特产 面向对象 项目优化 项目构建 黑洞
Your browser is out-of-date!

Update your browser to view this website correctly. Update my browser now

×