如何正確的使用WeakHashMap

這篇文章給大家介紹如何正確的使用WeakHashMap,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助。

創(chuàng)新互聯(lián)是一家網(wǎng)站設(shè)計(jì)公司,集創(chuàng)意、互聯(lián)網(wǎng)應(yīng)用、軟件技術(shù)為一體的創(chuàng)意網(wǎng)站建設(shè)服務(wù)商,主營(yíng)產(chǎn)品:成都響應(yīng)式網(wǎng)站建設(shè)、品牌網(wǎng)站制作、全網(wǎng)營(yíng)銷推廣。我們專注企業(yè)品牌在網(wǎng)站中的整體樹立,網(wǎng)絡(luò)互動(dòng)的體驗(yàn),以及在手機(jī)等移動(dòng)端的優(yōu)質(zhì)呈現(xiàn)。做網(wǎng)站、成都做網(wǎng)站、移動(dòng)互聯(lián)產(chǎn)品、網(wǎng)絡(luò)運(yùn)營(yíng)、VI設(shè)計(jì)、云產(chǎn)品.運(yùn)維為核心業(yè)務(wù)。為用戶提供一站式解決方案,我們深知市場(chǎng)的競(jìng)爭(zhēng)激烈,認(rèn)真對(duì)待每位客戶,為客戶提供賞析悅目的作品,網(wǎng)站的價(jià)值服務(wù)。

?在學(xué)習(xí)WeakHashMap時(shí)了解到,如果map里面的key只有map本身引用時(shí),就會(huì)將key對(duì)應(yīng)的Entry清除掉。查看WeakHashMap的源碼發(fā)現(xiàn),Entry繼承了WeakReference類,并且實(shí)例化Entry對(duì)象時(shí),所有的key都會(huì)通過調(diào)用super(key,queue)方法保存成對(duì)實(shí)際對(duì)象的弱引用。實(shí)際上,弱引用在構(gòu)造時(shí)也需要傳入一個(gè)對(duì)象的強(qiáng)引用作為參數(shù)。例如:

Car car = new Car(22000,"silver");
WeakReference<Car> weakCar = new WeakReference<Car>(car);

??HashMap和WeakHashMap的區(qū)別也在于此,HashMap的key是對(duì)實(shí)際對(duì)象的強(qiáng)引用。

??弱引用(WeakReference)的特性是:當(dāng)gc線程發(fā)現(xiàn)某個(gè)對(duì)象只有弱引用指向它,那么就會(huì)將其銷毀并回收內(nèi)存。WeakReference也會(huì)被加入到引用隊(duì)列queue中。

??理解了相關(guān)概念之后,對(duì)WeakHashMap的實(shí)際應(yīng)用感到很好奇。然后發(fā)現(xiàn)tomcat的源碼里,實(shí)現(xiàn)緩存時(shí)會(huì)用到WeakHashMap。

package org.apache.tomcat.util.collections;

import java.util.Map;
import java.util.WeakHashMap;
import java.util.concurrent.ConcurrentHashMap;

public final class ConcurrentCache<K,V> {

  private final int size;

  private final Map<K,V> eden;

  private final Map<K,V> longterm;

  public ConcurrentCache(int size) {
    this.size = size;
    this.eden = new ConcurrentHashMap<>(size);
    this.longterm = new WeakHashMap<>(size);
  }

  public V get(K k) {
    V v = this.eden.get(k);
    if (v == null) {
      synchronized (longterm) {
        v = this.longterm.get(k);
      }
      if (v != null) {
        this.eden.put(k, v);
      }
    }
    return v;
  }

  public void put(K k, V v) {
    if (this.eden.size() >= size) {
      synchronized (longterm) {
        this.longterm.putAll(this.eden);
      }
      this.eden.clear();
    }
    this.eden.put(k, v);
  }
}

??源碼中有eden和longterm的兩個(gè)map,對(duì)jvm堆區(qū)有所了解的話,可以猜測(cè)出tomcat在這里是使用ConcurrentHashMap和WeakHashMap做了分代的緩存。在put方法里,在插入一個(gè)k-v時(shí),先檢查eden緩存的容量是不是超了。沒有超就直接放入eden緩存,如果超了則鎖定longterm將eden中所有的k-v都放入longterm。再將eden清空并插入k-v。在get方法中,也是優(yōu)先從eden中找對(duì)應(yīng)的v,如果沒有則進(jìn)入longterm緩存中查找,找到后就加入eden緩存并返回。

??經(jīng)過這樣的設(shè)計(jì),相對(duì)常用的對(duì)象都能在eden緩存中找到,不常用(有可能被銷毀的對(duì)象)的則進(jìn)入longterm緩存。而longterm的key的實(shí)際對(duì)象沒有其他引用指向它時(shí),gc就會(huì)自動(dòng)回收heap中該弱引用指向的實(shí)際對(duì)象,弱引用進(jìn)入引用隊(duì)列。longterm調(diào)用expungeStaleEntries()方法,遍歷引用隊(duì)列中的弱引用,并清除對(duì)應(yīng)的Entry,不會(huì)造成內(nèi)存空間的浪費(fèi)。

關(guān)于如何正確的使用WeakHashMap就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,可以學(xué)到更多知識(shí)。如果覺得文章不錯(cuò),可以把它分享出去讓更多的人看到。

網(wǎng)站題目:如何正確的使用WeakHashMap
文章分享:http://bm7419.com/article2/psdjoc.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供App設(shè)計(jì)、網(wǎng)站排名、移動(dòng)網(wǎng)站建設(shè)、企業(yè)網(wǎng)站制作、關(guān)鍵詞優(yōu)化、動(dòng)態(tài)網(wǎng)站

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來源: 創(chuàng)新互聯(lián)

微信小程序開發(fā)