WeakHashMap 指南

WeakHashMap 指南

1. 概览

在这篇文章中,我们将会讨论 java.util包中的 WeakHashMap类。

为了能够理解 WeakHashMap这种数据结构,我们将会使用它来实现一个简单的缓存功能。然而,请谨记,该缓存功能仅仅是帮助理解该类的数据结构;自己去实现一个缓存往往并不是很值得的。

简单说, WeakHashMap是基于 hashtable的 ,对Map接口的实现,其 keyWeakReference类型。

在日常使用中,当 WeakHashMap中一个entry的key不再被使用的时候,该entry会被从 WeakHashMap中移除,意即没有指向那个key的单独的引用。当GC回收一个key的时候,该key所在的entry也会被很有效率的从map中移除。所以这个类的行为与其他 Map的实现还是存在一些不同的。

2. 强引用,软引用和弱引用

为了理解 WeakHashMap是如何工作的,需要了解一下 WeakReference这个类,这个类是 WeakHashMap中key的基本结构。在 JAVA中,有三个主要类型的引用,在下面的小节中会详细叙述。

2.1 强引用

强引用是我们日常编程中最常用的一种引用类型:

Integer prime = 1;

prime持有一个值为1的对象的强引用。任何对象,只要有一个强引用指向它,那么就不满足GC条件。

2.2 软引用

简单来说,一个对象,如果有一个软引用指向它,直到JVM需要内存的时候才会被GC。

下面是Java中创建一个软引用的代码片段:

Integer prime = 1;
SoftReference<Integer> soft = new SoftReference<Integer>(prime);
prime = null;

prime是指向对象的强引用。

接下来,将prime包装进入一个软引用。然后将强引用赋值为null,prime对象现在是满足GC条件的了,但是只有当JVM需要内存的时候才会被回收。

2.3 弱引用

只有被弱引用所持有的对象是对GC友好的(意即很容易被收集);在这种情况下,对于该对象的GC,发生时机不会延迟到JVM需要内存时。

下面是弱引用的一个例子:

Integer prime = 1;
WeakReference<Integer> weak = new WeakReference<>(prime);
prime = null;

当prime被赋值为null时,prime对象会在下个GC周期被回收,因为没有强引用指向它了。弱引用被用作 WeakHashMap中作为key的基本类型。

3. 使用 WeakHashMap 实现一个有效率的内存缓存功能

在缓存中,我们需要将一个较大的图片对象作为value,图片名字作为key。接下来,我们为这个问题选择一个合适的map实现。

简单的使用 HashMap不是一个很好的选择。因为value对象可能会占据很大的内存空间。更重要的是,他们从来不会被GC回收,尽管这些对象再也不会被使用了。

理想情况下,我们想要这样一种 Map的实现:允许GC自动删除回收不再使用的对象。当应用中的一个大图片对象的key不再被使用的时候,对应的entry会被从内存中删除。

幸运的是, WeakHashMap恰好具备这些特性。下面看一下:

WeakHashMap<UniqueImageName, BigImage> map = new WeakHashMap<>();
BigImage bigImage = new BigImage("image_id");
UniqueImageName imageName = new UniqueImageName("name_of_big_image");
 
map.put(imageName, bigImage);
assertTrue(map.containsKey(imageName));
 
imageName = null;
System.gc();
 
await().atMost(10, TimeUnit.SECONDS).until(map::isEmpty);

我们创建一个 WeakHashMap对象用来存储 BigImage对象。将 BigImage作为值,一个 imageName对象作为key。 imageName以 **WeakReference **的类型存在。

接下来,设置imageName 为null,这样就没有任何引用指向 bigImage 对象了。 WeakReference 的默认行为是会在下一个GC中将entry回收,因此在下一个GC过程中, 这个entry会被删除。

调用 System.gc() 强制JVM触发GC,在GC之后, WeakHashMap 会变为空:

WeakHashMap<UniqueImageName, BigImage> map = new WeakHashMap<>();
BigImage bigImageFirst = new BigImage("foo");
UniqueImageName imageNameFirst = new UniqueImageName("name_of_big_image");
 
BigImage bigImageSecond = new BigImage("foo_2");
UniqueImageName imageNameSecond = new UniqueImageName("name_of_big_image_2");
 
map.put(imageNameFirst, bigImageFirst);
map.put(imageNameSecond, bigImageSecond);
  
assertTrue(map.containsKey(imageNameFirst));
assertTrue(map.containsKey(imageNameSecond));
 
imageNameFirst = null;
System.gc();
 
await().atMost(10, TimeUnit.SECONDS)
  .until(() -> map.size() == 1);
await().atMost(10, TimeUnit.SECONDS)
  .until(() -> map.containsKey(imageNameSecond));

只有 imageNameFirst 引用被设置为null。imageNameSecond 并没有改变,在GC之后, map将只会持有一个entry--------imageNameSecond

4. 总结

在这篇文章中,我们学习了JAVA重点引用类型,并完全理解了WeakHashMap 是如何工作的。我们利用***WeakHashMap ***创建了一个简单的缓存并且测试了他是否能够预期的进行工作。


©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容

  • WeakHashMap WeakHashMap介绍 WeakHashMap继承于AbstractMap,实现了Ma...
    史路比阅读 838评论 0 0
  • 概述   在学习WeakHashMap之前,先简单来说一下Java中的4种引用类型,它们分别是:强引用(Stron...
    骑着乌龟去看海阅读 734评论 2 5
  • 感知GC。怎么感知:* 通过get来判断已经被GC(PhantomReference 在任何时候get都是null...
    YDDMAX_Y阅读 1,826评论 0 4
  • 作者:周明耀原文地址:http://www.ibm.com/developerworks/cn/java/j-lo...
    IT程序狮阅读 662评论 0 5
  • 风起了 也该是时候起风了 单车少年 仍是不肯踏上叶的遗骸 白雾依旧是笼罩天地的裹尸布 背负残萎死叶的大道 依稀地发...
    良人儿2阅读 149评论 0 0