Java LocalCache 本地緩存的實現(xiàn)

使用場景
在Java應(yīng)用中喊暖,對于訪問頻率高刽射,更新少的數(shù)據(jù)军拟,通常的方案是將這類數(shù)據(jù)加入緩存中。相對從數(shù)據(jù)庫中讀取來說誓禁,讀緩存效率會有很大提升懈息。
在集群環(huán)境下,常用的分布式緩存有Redis摹恰、Memcached等辫继。但在某些業(yè)務(wù)場景上,可能不需要去搭建一套復(fù)雜的分布式緩存系統(tǒng)俗慈,在單機環(huán)境下姑宽,通常是會希望使用內(nèi)部的緩存(LocalCache)。
實現(xiàn)
這里提供了兩種LocalCache的實現(xiàn)闺阱,一種是基于ConcurrentHashMap實現(xiàn)基本本地緩存炮车,另外一種是基于LinkedHashMap實現(xiàn)LRU策略的本地緩存。
基于ConcurrentHashMap的實現(xiàn)

    static {
        timer = new Timer();
        map = new ConcurrentHashMap<>();
    }

以ConcurrentHashMap作為緩存的存儲結(jié)構(gòu)酣溃。因為ConcurrentHashMap的線程安全的瘦穆,所以基于此實現(xiàn)的LocalCache在多線程并發(fā)環(huán)境的操作是安全的。在JDK1.8中赊豌,ConcurrentHashMap是支持完全并發(fā)讀扛或,這對本地緩存的效率也是一種提升。通過調(diào)用ConcurrentHashMap對map的操作來實現(xiàn)對緩存的操作碘饼。
私有構(gòu)造函數(shù)

    private LocalCache() {

    }

LocalCache是工具類熙兔,通過私有構(gòu)造函數(shù)強化不可實例化的能力悲伶。
緩存清除機制

   /**
     * 清除緩存任務(wù)類
     */
    static class CleanWorkerTask extends TimerTask {

        private String key;

        public CleanWorkerTask(String key) {
            this.key = key;
        }

        public void run() {
            LocalCache.remove(key);
        }
    }

清理失效緩存是由Timer類實現(xiàn)的。內(nèi)部類CleanWorkerTask繼承于TimerTask用戶清除緩存住涉。每當新增一個元素的時候麸锉,都會調(diào)用timer.schedule加載清除緩存的任務(wù)。
基于LinkedHashMap的實現(xiàn)
以LinkedHashMap作為緩存的存儲結(jié)構(gòu)舆声。主要是通過LinkedHashMap的按照訪問順序的特性來實現(xiàn)LRU策略淮椰。
LRU
LRU是Least Recently Used的縮寫,即最近最久未使用纳寂。LRU緩存將會利用這個算法來淘汰緩存中老的數(shù)據(jù)元素,從而優(yōu)化內(nèi)存空間泻拦。
基于LRU策略的map
這里利用LinkedHashMap來實現(xiàn)基于LRU策略的map毙芜。通過調(diào)用父類LinkedHashMap的構(gòu)造函數(shù)來實例化map。參數(shù)accessOrder設(shè)置為true保證其可以實現(xiàn)LRU策略争拐。

static class LRUMap<K, V> extends LinkedHashMap<K, V> {

        ...  // 省略部分代碼
        
        public LRUMap(int initialCapacity, float loadFactor) {
            super(initialCapacity, loadFactor, true);
        }

        ... // 省略部分代碼
        
        /**
         * 重寫LinkedHashMap中removeEldestEntry方法;
         * 新增元素的時候,會判斷當前map大小是否超過DEFAULT_MAX_CAPACITY,超過則移除map中最老的節(jié)點;
         *
         * @param eldest
         * @return
         */
        protected boolean removeEldestEntry(Map.Entry<K, V> eldest) {
            return size() > DEFAULT_MAX_CAPACITY;
        }

    }

線程安全

        /**
         * 讀寫鎖
         */
        private final ReadWriteLock readWriteLock = new ReentrantReadWriteLock();

        private final Lock rLock = readWriteLock.readLock();

        private final Lock wLock = readWriteLock.writeLock();

LinkedHashMap并不是線程安全腋粥,如果不加控制的在多線程環(huán)境下使用的話,會有問題架曹。所以在LRUMap中引入了ReentrantReadWriteLock讀寫鎖隘冲,來控制并發(fā)問題。
緩存淘汰機制

        protected boolean removeEldestEntry(Map.Entry<K, V> eldest) {
            return size() > DEFAULT_MAX_CAPACITY;
        }

此處重寫LinkedHashMap中removeEldestEntry方法绑雄, 當緩存新增元素的時候,會判斷當前map大小是否超過DEFAULT_MAX_CAPACITY,超過則移除map中最老的節(jié)點展辞。
緩存清除機制
緩存清除機制與ConcurrentHashMap的實現(xiàn)一致,均是通過timer實現(xiàn)万牺。
文章來源于網(wǎng)絡(luò)罗珍。
感謝大家閱讀,歡迎大家私信討論脚粟。給大家推薦一個Java技術(shù)交流群:473984645里面會分享一些資深架構(gòu)師錄制的視頻資料:有Spring覆旱,MyBatis,Netty源碼分析核无,高并發(fā)扣唱、高性能、分布式团南、微服務(wù)架構(gòu)的原理噪沙,JVM性能優(yōu)化、分布式架構(gòu)等這些成為架構(gòu)師必備的知識體系已慢。還能領(lǐng)取免費的學習資源曲聂,目前受益良多!
推薦大家閱讀:
Java高級架構(gòu)學習資料分享+架構(gòu)師成長之路?
個人整理了更多資料以PDF文件的形式分享給大家佑惠,需要查閱的程序員朋友可以來免費領(lǐng)取朋腋。還有我的學習筆記PDF文件也免費分享給有需要朋友齐疙!

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市旭咽,隨后出現(xiàn)的幾起案子贞奋,更是在濱河造成了極大的恐慌,老刑警劉巖穷绵,帶你破解...
    沈念sama閱讀 219,427評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件轿塔,死亡現(xiàn)場離奇詭異,居然都是意外死亡仲墨,警方通過查閱死者的電腦和手機勾缭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,551評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來目养,“玉大人俩由,你說我怎么就攤上這事“┮希” “怎么了幻梯?”我有些...
    開封第一講書人閱讀 165,747評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長努释。 經(jīng)常有香客問我碘梢,道長,這世上最難降的妖魔是什么伐蒂? 我笑而不...
    開封第一講書人閱讀 58,939評論 1 295
  • 正文 為了忘掉前任煞躬,我火速辦了婚禮,結(jié)果婚禮上逸邦,老公的妹妹穿的比我還像新娘汰翠。我一直安慰自己,他們只是感情好昭雌,可當我...
    茶點故事閱讀 67,955評論 6 392
  • 文/花漫 我一把揭開白布复唤。 她就那樣靜靜地躺著,像睡著了一般烛卧。 火紅的嫁衣襯著肌膚如雪佛纫。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,737評論 1 305
  • 那天总放,我揣著相機與錄音呈宇,去河邊找鬼。 笑死局雄,一個胖子當著我的面吹牛甥啄,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播炬搭,決...
    沈念sama閱讀 40,448評論 3 420
  • 文/蒼蘭香墨 我猛地睜開眼蜈漓,長吁一口氣:“原來是場噩夢啊……” “哼穆桂!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起融虽,我...
    開封第一講書人閱讀 39,352評論 0 276
  • 序言:老撾萬榮一對情侶失蹤享完,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后有额,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體般又,經(jīng)...
    沈念sama閱讀 45,834評論 1 317
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,992評論 3 338
  • 正文 我和宋清朗相戀三年巍佑,在試婚紗的時候發(fā)現(xiàn)自己被綠了茴迁。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,133評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡萤衰,死狀恐怖笋熬,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情腻菇,我是刑警寧澤,帶...
    沈念sama閱讀 35,815評論 5 346
  • 正文 年R本政府宣布昔馋,位于F島的核電站筹吐,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏秘遏。R本人自食惡果不足惜丘薛,卻給世界環(huán)境...
    茶點故事閱讀 41,477評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望邦危。 院中可真熱鬧洋侨,春花似錦、人聲如沸倦蚪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,022評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽陵且。三九已至裁僧,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間慕购,已是汗流浹背聊疲。 一陣腳步聲響...
    開封第一講書人閱讀 33,147評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留沪悲,地道東北人获洲。 一個月前我還...
    沈念sama閱讀 48,398評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像殿如,于是被迫代替她去往敵國和親贡珊。 傳聞我的和親對象是個殘疾皇子最爬,可洞房花燭夜當晚...
    茶點故事閱讀 45,077評論 2 355

推薦閱讀更多精彩內(nèi)容