緩存雪崩導(dǎo)致的危害和解決辦法

1. Redis 數(shù)據(jù)失效導(dǎo)致的雪崩

因?yàn)?strong>緩存失效李滴,從而導(dǎo)致大量請求導(dǎo)向數(shù)據(jù)庫呢蛤。

  1. 大量請求黑毅,導(dǎo)致數(shù)據(jù)庫處理不過來祭衩,整個(gè)系統(tǒng)依賴數(shù)據(jù)庫的功能全部崩潰
  2. 單系統(tǒng)掛掉灶体,其他依賴于該系統(tǒng)的應(yīng)用也會(huì)出現(xiàn)不穩(wěn)定甚至崩潰

2. Redis數(shù)據(jù)失效的場景

  • 最大內(nèi)存控制
    maxmemory 最大內(nèi)存閾值
    maxmemory-policy 到達(dá)閾值的執(zhí)行策略

3. 緩存雪崩解決方案

3.1 Semaphore信號量限流
  • J.U.C包重要的并發(fā)編程工具類
    又稱“信號量”,控制多個(gè)線程爭搶許可汪厨。

核心方法

  • acquire:獲取一個(gè)許可赃春,如果沒有就等待愉择,
  • release:釋放一個(gè)許可劫乱。
  • 典型場景∶
    1织中、代碼并發(fā)處理限流;

  • 例子

package cn.lazyfennec.cache.redis.service;

import cn.lazyfennec.cache.redis.annotations.NeteaseCache;
import cn.lazyfennec.cache.redis.dao.UserDao;
import cn.lazyfennec.cache.redis.pojo.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.cache.annotation.CacheEvict;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.jdbc.core.JdbcTemplate;
import org.springframework.stereotype.Service;

import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.Semaphore;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.locks.ReentrantLock;

@Service // 默認(rèn) 單實(shí)例
public class UserService2 {

    @Autowired
    UserDao userDao;

    @Autowired
    RedisTemplate redisTemplate; // spring提供的一個(gè)redis客戶端,底層封裝了jedis等客戶端

    // userId ---> lock 記錄每一個(gè)userId當(dāng)前的查詢情況
    static Map<String, ReentrantLock> mapLock = new ConcurrentHashMap<>();

    static Semaphore semaphore = new Semaphore(50); // 信號量 50 -- 類似車票

    /**
     * 根據(jù)ID查詢用戶信息 (redis緩存衷戈,用戶信息以json字符串格式存在(序列化))
     */
    public User findUserById(String userId) throws Exception {

        // 1. 先讀取緩存
        Object cacheValue = redisTemplate.opsForValue().get(userId); // redisTemplate是spring提供的redis客戶端
        if (cacheValue != null) {
            System.out.println("###緩存命中:" + ((User) cacheValue).getUname());
            return (User) cacheValue;
        }

        // ---------------緩存miss之后流程--------------
        ReentrantLock reentrantLock = new ReentrantLock();
        try {
            if (mapLock.putIfAbsent(userId, reentrantLock) != null) { // 有返回值代表存在鎖
                reentrantLock = mapLock.get(userId);
            }
            Thread.sleep(3000);// TODO 停頓3秒狭吼,等下一個(gè)線程過來,模擬多個(gè)用戶同時(shí)并發(fā)請求的場景
            reentrantLock.lock(); // 爭搶鎖,搶不到的排隊(duì)---1個(gè)請求查詢數(shù)據(jù)庫 --- 599個(gè)等待
            Thread.sleep(3000);// TODO 停頓3秒殖妇,模擬lock獲取之后業(yè)務(wù)處理時(shí)間

            // 再次查詢緩存 -- 避免大量重復(fù)數(shù)據(jù)庫查詢
            cacheValue = redisTemplate.opsForValue().get(userId); // redisTemplate是spring提供的redis客戶端
            if (cacheValue != null) {
                System.out.println("###緩存命中:" + ((User) cacheValue).getUname());
                return (User) cacheValue;
            }

            semaphore.acquire(); // 獲取信號量 刁笙,沒有獲取到

            // 2. 如果緩存miss,則查詢數(shù)據(jù)庫
            User user = userDao.findUserById(userId);
            System.out.println("***緩存miss:" + user.getUname());
            // 3. 設(shè)置緩存(重建緩存) // 主播信息查詢緩存
            redisTemplate.opsForValue().set(userId, user);// set key value
            redisTemplate.expire(userId, 100, TimeUnit.SECONDS); // 需要手動(dòng)設(shè)

            semaphore.release(); // 釋放信號量

            return user;
        } finally {
            if (!reentrantLock.hasQueuedThreads()) { // 當(dāng)鎖最后一個(gè)釋放的時(shí)候谦趣,刪除掉
                mapLock.remove(userId);
            }
            reentrantLock.unlock();
        }

    }


    @CacheEvict(value = "user", key = "#user.uid") // 方法執(zhí)行結(jié)束疲吸,清除緩存
    public void updateUser(User user) {
        String sql = "update tb_user_base set uname = ? where uid=?";
        jdbcTemplate.update(sql, new String[]{user.getUname(), user.getUid()});
    }

    /**
     * 根據(jù)ID查詢用戶名稱
     */
    // 我自己實(shí)現(xiàn)一個(gè)類似的注解
    @NeteaseCache(value = "uname", key = "#userId") // 緩存
    public String findUserNameById(String userId) {
        // 查詢數(shù)據(jù)庫
        String sql = "select uname from tb_user_base where uid=?";
        String uname = jdbcTemplate.queryForObject(sql, new String[]{userId}, String.class);

        return uname;
    }

    @Autowired
    JdbcTemplate jdbcTemplate; // spring提供jdbc一個(gè)工具(mybastis類似)
}
3.2 容錯(cuò)降級

如果覺得有收獲就點(diǎn)個(gè)贊吧,更多知識前鹅,請點(diǎn)擊關(guān)注查看我的主頁信息哦~

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末摘悴,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子舰绘,更是在濱河造成了極大的恐慌蹂喻,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,265評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件捂寿,死亡現(xiàn)場離奇詭異口四,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)秦陋,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,078評論 2 385
  • 文/潘曉璐 我一進(jìn)店門蔓彩,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人踱侣,你說我怎么就攤上這事粪小。” “怎么了抡句?”我有些...
    開封第一講書人閱讀 156,852評論 0 347
  • 文/不壞的土叔 我叫張陵探膊,是天一觀的道長。 經(jīng)常有香客問我待榔,道長逞壁,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,408評論 1 283
  • 正文 為了忘掉前任锐锣,我火速辦了婚禮腌闯,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘雕憔。我一直安慰自己姿骏,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,445評論 5 384
  • 文/花漫 我一把揭開白布斤彼。 她就那樣靜靜地躺著分瘦,像睡著了一般蘸泻。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上嘲玫,一...
    開封第一講書人閱讀 49,772評論 1 290
  • 那天悦施,我揣著相機(jī)與錄音,去河邊找鬼去团。 笑死抡诞,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的土陪。 我是一名探鬼主播昼汗,決...
    沈念sama閱讀 38,921評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼鬼雀!你這毒婦竟也來了乔遮?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,688評論 0 266
  • 序言:老撾萬榮一對情侶失蹤取刃,失蹤者是張志新(化名)和其女友劉穎蹋肮,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體璧疗,經(jīng)...
    沈念sama閱讀 44,130評論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡坯辩,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,467評論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了崩侠。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片漆魔。...
    茶點(diǎn)故事閱讀 38,617評論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖却音,靈堂內(nèi)的尸體忽然破棺而出改抡,到底是詐尸還是另有隱情,我是刑警寧澤系瓢,帶...
    沈念sama閱讀 34,276評論 4 329
  • 正文 年R本政府宣布阿纤,位于F島的核電站,受9級特大地震影響夷陋,放射性物質(zhì)發(fā)生泄漏欠拾。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,882評論 3 312
  • 文/蒙蒙 一骗绕、第九天 我趴在偏房一處隱蔽的房頂上張望藐窄。 院中可真熱鬧,春花似錦酬土、人聲如沸荆忍。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,740評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽刹枉。三九已至践惑,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間嘶卧,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 31,967評論 1 265
  • 我被黑心中介騙來泰國打工凉袱, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留芥吟,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,315評論 2 360
  • 正文 我出身青樓专甩,卻偏偏與公主長得像钟鸵,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子涤躲,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,486評論 2 348

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