亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

java使用軟引用實現(xiàn)緩存機制示例

 更新時間:2022年08月07日 09:17:03   作者:My2538772270  
這篇文章主要為大家介紹了java使用軟引用實現(xiàn)緩存機制示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪

正文

“讀多寫少”是大部分項目的一個特點。例如“購物”,總是看的人多(讀)、買的人少(寫)。因此,如果能減少“讀”請求的次數(shù),就能減少服務(wù)端的壓力。最直接的減少“讀”請求次數(shù)的方法就是使用緩存。

軟引用和強引用

對于同一個讀請求,只需要在第一次訪問時從數(shù)據(jù)庫中查詢數(shù)據(jù),并將查詢到的數(shù)據(jù)保存到緩存中,之后的查詢請求就可以直接在緩存中獲取,從而減少對數(shù)據(jù)庫的訪問次數(shù)。

這種情況我們生活種經(jīng)常會看到,比如訪問某app某商品,第一次進去會加載一會會,后面繼續(xù)點擊是直接出現(xiàn)。

根據(jù)目前所學(xué)知識,我們可以使用 HashMap 在內(nèi)存級別實現(xiàn)緩存功能。

例如,可以使用一個 HashMap 對象保存客戶端第一次請求的結(jié)果,之后,當(dāng)客戶端再次發(fā)起讀請求時,就從 HashMap 對象中遍歷查詢,如果 HashMap 中已經(jīng)保存過客戶要查詢的數(shù)據(jù),就直接返回,否則再向數(shù)據(jù)庫發(fā)起查詢請求,并將查詢結(jié)果保存到 HashMap 中。

這種緩存的設(shè)計思路十分簡單,但也存在一個問題:HashMap 中緩存的數(shù)據(jù)何時被清空?

內(nèi)存容量是有限制的,如果永無止盡的向 HashMap 緩存數(shù)據(jù),顯然會對內(nèi)存容量帶來壓力。一種解決方案就是使用 JVM 提供的軟引用,實現(xiàn)對 HashMap 中緩存數(shù)據(jù)的淘汰策略。

開發(fā)中最常使用的是強引用,例如 Goods goods = new Goods() 就創(chuàng)建了一個強引用對象“goods”。只要強引用的作用域沒有結(jié)束,或者沒有被開發(fā)者手工設(shè)置為 null,那么強引用對象會始終存在于 JVM 內(nèi)存中。

而 JVM 提供的軟引用就比較靈活:當(dāng) JVM 的內(nèi)存足夠時,GC 對待軟引用和強引用的方式是一樣的;但當(dāng) JVM 的內(nèi)存不足時,GC 就會去主動回收軟引用對象。

可見,非常適合將緩存的對象存放在軟引用中。軟引用需要借助 JDK 提供的 java.lang.ref.SoftReference 類來實現(xiàn)。

項目

使用idea創(chuàng)建一個maven項目

結(jié)構(gòu)如下

首先對Good實體類進行編寫。

要求,goods有屬性id,name并書寫他的getset方法,以及有參無參構(gòu)造器。

這里代碼省略。

然后我們在goodbase里面編寫代碼,模擬一個數(shù)據(jù)庫

里面主要有hashmap,并且通過get方法,得到該hashmap

public class GoodsBase {
    private static Map<String, Goods> base = new HashMap<>();
    public static Map<String, Goods> getBase() {
        return base;
    }
}

然后書寫goodscache緩存類

這里我們需要接觸一個新關(guān)鍵字volatile

  • 使用volatile關(guān)鍵字會強制將修改的值立即寫入主存;
  • 使用volatile關(guān)鍵字的話,當(dāng)主線程修改時,會導(dǎo)致RunThread的工作內(nèi)存中isRunning變量的緩存值變得無效。
  • 由于RunThread的工作內(nèi)存中緩存變量isRunning緩存無效,所以會再次從主存中讀取isRunning變量值。

在map里面通過泛型把緩存對象存儲在軟引用里面(map里面)

代碼如下:

public class GoodsCache {
    private volatile static GoodsCache goodsCache;
    public GoodsCache(){
        this.cache = new HashMap<>();
    }
    public static GoodsCache getGoodsCache() {
        if(goodsCache == null) {
            synchronized (GoodsCache.class){
                if(goodsCache == null){
                    goodsCache = new GoodsCache();
                }
            }
        }
        return goodsCache;
    }
    // 將緩存對象存儲在軟引用中
    private Map<String, SoftReference<Goods>> cache;
    // 根據(jù)id存儲緩存Goods對象
    public void setCache(Goods goods) {
        cache.put(goods.getId(), new SoftReference<Goods>(goods));
        System.out.println("添加數(shù)據(jù)到緩存成功");
    }
    // 根據(jù)id從緩存中獲取對象
    public Goods getCache(String id) {
        // 根據(jù)id,獲取緩存對象的軟引用
        SoftReference<Goods> softRef = cache.get(id);
        return softRef == null ? null : softRef.get();
    }
    public void delCache(String id) {
        cache.remove(id);
        System.out.println("從緩存刪除數(shù)據(jù)成功");
    }
}

goodsservice模擬數(shù)據(jù)庫增刪改查

接下來我們書寫goodsservice代碼,來模擬數(shù)據(jù)庫增刪改查,不過我們是通過id來進行

public class GoodsService {
    GoodsCache goodsCache = GoodsCache.getGoodsCache();
    public Goods getById(String id){
        if(goodsCache.getCache(id) == null){
            Goods goods = GoodsBase.getBase().get(id);
            goodsCache.setCache(goods);
            System.out.println("從數(shù)據(jù)庫讀取數(shù)據(jù)");
            System.out.println(goods.getName());
            return goods;
        }
        System.out.println(goodsCache.getCache(id).getName());
        return goodsCache.getCache(id);
    }
    public void add(Goods goods){
        goodsCache.setCache(goods);
        GoodsBase.getBase().put(goods.getId(), goods);
        System.out.println("添加數(shù)據(jù)到數(shù)據(jù)庫");
    }
    public void deleteById(String id){
        if(goodsCache.getCache(id) != null){
            goodsCache.delCache(id);
        }
        GoodsBase.getBase().remove(id);
    }
}

最后我們書寫test文件

運行結(jié)果

可以看到第二次運行 goodsService.getById("1"); 是從緩存中直接讀取的數(shù)據(jù),也可以看出,其實用軟引用實現(xiàn)緩存機制,讀取的對象是同一個對象。

以上就是java使用軟引用實現(xiàn)緩存機制示例的詳細內(nèi)容,更多關(guān)于java軟引用緩存機制的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • Java(TM) Platform SE binary 打開jar文件的操作

    Java(TM) Platform SE binary 打開jar文件的操作

    這篇文章主要介紹了Java(TM) Platform SE binary 打開jar文件的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-02-02
  • SpringBoot使用mybatis步驟總結(jié)

    SpringBoot使用mybatis步驟總結(jié)

    今天給大家?guī)淼氖顷P(guān)于Java的相關(guān)知識,文章圍繞著SpringBoot使用mybatis步驟展開,文中有非常詳細的介紹及代碼示例,需要的朋友可以參考下
    2021-06-06
  • Java 騰訊驗證碼平臺使用實例

    Java 騰訊驗證碼平臺使用實例

    這篇文章主要介紹了Java 騰訊驗證碼平臺使用實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2020-02-02
  • Java中ArrayList的使用方法簡單介紹

    Java中ArrayList的使用方法簡單介紹

    這篇文章主要為大家簡單介紹了Java中ArrayList的使用方法,針對ArrayList去重問題進行擴展分析,感興趣的小伙伴們可以參考一下
    2016-07-07
  • JAVA動態(tài)維度笛卡爾積輸出的實現(xiàn)

    JAVA動態(tài)維度笛卡爾積輸出的實現(xiàn)

    本文主要介紹了JAVA動態(tài)維度笛卡爾積輸出的實現(xiàn),通過動態(tài)生成笛卡爾積,可以方便地處理多維數(shù)據(jù)集,提高數(shù)據(jù)處理效率,具有一定的參考價值,感興趣的可以了解一下
    2024-02-02
  • Java通過正則表達式獲取域名簡單示例

    Java通過正則表達式獲取域名簡單示例

    在Java中可以使用正則表達式來從字符串中匹配和提取域名,下面這篇文章主要給大家介紹了關(guān)于Java通過正則表達式獲取域名的相關(guān)資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2023-12-12
  • 在IDEA中搭建最小可用SpringMVC項目(純Java配置)

    在IDEA中搭建最小可用SpringMVC項目(純Java配置)

    這篇文章主要介紹了在IDEA中搭建最小可用SpringMVC項目(純Java配置),文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2019-12-12
  • Spring中@PropertySource注解使用場景解析

    Spring中@PropertySource注解使用場景解析

    這篇文章主要介紹了Spring中@PropertySource注解使用場景解析,@PropertySource注解就是Spring中提供的一個可以加載配置文件的注解,并且可以將配置文件中的內(nèi)容存放到Spring的環(huán)境變量中,需要的朋友可以參考下
    2023-11-11
  • 使用Intellij IDEA查看Java源碼技巧

    使用Intellij IDEA查看Java源碼技巧

    這篇文章主要介紹了使用Intellij IDEA查看Java源碼技巧,本文通過圖文并茂的形式給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-04-04
  • 小程序與后端Java接口交互實現(xiàn)HelloWorld入門

    小程序與后端Java接口交互實現(xiàn)HelloWorld入門

    本文主要介紹了小程序與后端Java接口交互實現(xiàn)HelloWorld入門 ,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-07-07

最新評論