欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

springboot中redis的緩存穿透問(wèn)題實(shí)現(xiàn)

 更新時(shí)間:2021年02月05日 08:57:04   作者:quintan  
這篇文章主要介紹了springboot中redis的緩存穿透問(wèn)題實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧

什么是緩存穿透問(wèn)題??

我們使用redis是為了減少數(shù)據(jù)庫(kù)的壓力,讓盡量多的請(qǐng)求去承壓能力比較大的redis,而不是數(shù)據(jù)庫(kù)。但是高并發(fā)條件下,可能會(huì)在redis還沒(méi)有緩存的時(shí)候,大量的請(qǐng)求同時(shí)進(jìn)入,導(dǎo)致一大批的請(qǐng)求直奔數(shù)據(jù)庫(kù),而不會(huì)經(jīng)過(guò)redis。使用代碼模擬緩存穿透問(wèn)題如下:

首先是service里面的代碼:

@Service
public class NewsService {
  @Autowired
  private NewsDAO newsDAO;

  //springboot自動(dòng)初始化,不需要我們進(jìn)行配置,直接注入到代碼中使用
  @Autowired
  private RedisTemplate<Object,Object> redisTemplate;

  public /*synchronized*/ List<News> getLatestNews(int userId,int offset,int limit){

    //設(shè)置序列化方式,防止亂碼
    redisTemplate.setKeySerializer(new StringRedisSerializer());

    //第一步:查詢緩存
    News news= (News) redisTemplate.opsForValue().get("newsKey");
    //判斷是否存在緩存
    if(null == news){//查詢數(shù)據(jù)庫(kù)
        news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
        //
        redisTemplate.opsForValue().set("newsKey",news);

        System.out.println("進(jìn)入數(shù)據(jù)庫(kù)。。。。。。。。");
      
    }else{
      System.out.println("進(jìn)入緩存。。。。。。。。。");
    }
    return newsDAO.selectByUserIdAndOffset(userId,offset,limit);

  }
}

然后是使用線程池在Controller里面對(duì)請(qǐng)求進(jìn)行模擬:

@Controller
public class HomeController {
  @Autowired
  UserService userService;

  @Autowired
  NewsService newsService;

  //遇到的坑,如果不加method,頁(yè)面啟動(dòng)不起來(lái)。
  @RequestMapping(value = "/home",method = {RequestMethod.GET, RequestMethod.POST})
  @ResponseBody
  public String index(Model model){
    //這邊是可以讀出數(shù)據(jù)來(lái)的

    //線程池------緩存穿透問(wèn)題的復(fù)現(xiàn)
    ExecutorService executorService = Executors.newFixedThreadPool(8*2);

    for(int i = 0;i < 50000;i++){
      executorService.submit(new Runnable() {
        @Override
        public void run() {
          List<News> newsList = newsService.getLatestNews(0,0,10);
        }
      });
    }

    List<News> newsList = newsService.getLatestNews(0,0,10);
    News news=newsList.get(0);
    return news.getImage();
  }
}

結(jié)果如圖:大量的請(qǐng)求進(jìn)入數(shù)據(jù)庫(kù),那么如何解決這個(gè)問(wèn)題?

方法一、在方法上加鎖:

@Service
public class NewsService {
  @Autowired
  private NewsDAO newsDAO;

  //springboot自動(dòng)初始化,不需要我們進(jìn)行配置,直接注入到代碼中使用
  @Autowired
  private RedisTemplate<Object,Object> redisTemplate;

  //第一種方式:方法加鎖
  public synchronized List<News> getLatestNews(int userId,int offset,int limit){

    //設(shè)置序列化方式,防止亂碼
    redisTemplate.setKeySerializer(new StringRedisSerializer());

    //第一步:查詢緩存
    News news= (News) redisTemplate.opsForValue().get("newsKey");
    //判斷是否存在緩存
    if(null == news){
//查詢數(shù)據(jù)庫(kù)
        news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
        //
        redisTemplate.opsForValue().set("newsKey",news);

        System.out.println("進(jìn)入數(shù)據(jù)庫(kù)。。。。。。。。");

    }else{
      System.out.println("進(jìn)入緩存。。。。。。。。。");
    }


    return newsDAO.selectByUserIdAndOffset(userId,offset,limit);

  }
}

 直接在方法上加鎖,保證每次只有一個(gè)請(qǐng)求可以進(jìn)入。但是這個(gè)方法存在一個(gè)缺陷,每次只有一個(gè)請(qǐng)求可以進(jìn)入,請(qǐng)求處理的速度變得相當(dāng)?shù)穆焕谙到y(tǒng)的實(shí)時(shí)性。

方法二、使用雙重校驗(yàn)鎖:

@Service
public class NewsService {
  @Autowired
  private NewsDAO newsDAO;

  //springboot自動(dòng)初始化,不需要我們進(jìn)行配置,直接注入到代碼中使用
  @Autowired
  private RedisTemplate<Object,Object> redisTemplate;

  //第一種方式:方法加鎖
  public /*synchronized*/ List<News> getLatestNews(int userId,int offset,int limit){

    //設(shè)置序列化方式,防止亂碼
    redisTemplate.setKeySerializer(new StringRedisSerializer());

    //第一步:查詢緩存
    News news= (News) redisTemplate.opsForValue().get("newsKey");
    //判斷是否存在緩存
    if(null == news){

      //第二種方式:雙重檢測(cè)鎖
      synchronized (this){
        //查詢數(shù)據(jù)庫(kù)
        news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
        //
        redisTemplate.opsForValue().set("newsKey",news);

        System.out.println("進(jìn)入數(shù)據(jù)庫(kù)。。。。。。。。");
      }

    }else{
      System.out.println("進(jìn)入緩存。。。。。。。。。");
    }


    return newsDAO.selectByUserIdAndOffset(userId,offset,limit);

  }
}

這個(gè)方法比較好,雖然不能保證只有一個(gè)請(qǐng)求請(qǐng)求數(shù)據(jù)庫(kù),但是當(dāng)?shù)谝慌?qǐng)求進(jìn)來(lái),第二批之后的所有請(qǐng)求全部會(huì)在緩存取數(shù)據(jù)。

到此這篇關(guān)于springboot中redis的緩存穿透問(wèn)題實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)springboot redis緩存穿透內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 利用Java實(shí)現(xiàn)和可被K整除的子數(shù)組完整實(shí)例

    利用Java實(shí)現(xiàn)和可被K整除的子數(shù)組完整實(shí)例

    這篇文章主要給大家介紹了關(guān)于利用Java實(shí)現(xiàn)和可被K整除的子數(shù)組的相關(guān)資料,這道題來(lái)自力扣,通過(guò)學(xué)習(xí)這道題的解題思路以及代碼對(duì)大家的學(xué)習(xí)或者工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2024-01-01
  • SpringBoot任意版本集成Swagger各種版本的操作指南

    SpringBoot任意版本集成Swagger各種版本的操作指南

    在學(xué)習(xí)Swagger生成API文檔的時(shí)候經(jīng)常會(huì)遇到問(wèn)題,而目前市面上大部分技術(shù)分享者的SpringBoot版本并沒(méi)和我們的同步,導(dǎo)致一些一模一樣的代碼,在我們的項(xiàng)目上卻無(wú)法使用,這是一個(gè)經(jīng)常性的問(wèn)題,本文章就旨在和大家搞定SpringBoot任意版本集成Swagger各種版本
    2024-07-07
  • Mybatis 中 Oracle 的拼接模糊查詢及用法詳解

    Mybatis 中 Oracle 的拼接模糊查詢及用法詳解

    這篇文章主要介紹了Mybatis 中 Oracle 的拼接模糊查詢及用法,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2018-08-08
  • Java源碼解析之詳解ReentrantLock

    Java源碼解析之詳解ReentrantLock

    今天給大家?guī)?lái)的是關(guān)于Java并發(fā)的相關(guān)知識(shí),文章圍繞著ReentrantLock源碼展開,文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下
    2021-06-06
  • java如何發(fā)送get請(qǐng)求獲取數(shù)據(jù)(附代碼)

    java如何發(fā)送get請(qǐng)求獲取數(shù)據(jù)(附代碼)

    這篇文章主要給大家介紹了關(guān)于java如何發(fā)送get請(qǐng)求獲取數(shù)據(jù)的相關(guān)資料,Java中的GET請(qǐng)求方法是HTTP協(xié)議中的一種請(qǐng)求方式,用于向服務(wù)器請(qǐng)求獲取資源,需要的朋友可以參考下
    2023-10-10
  • SpringBoot web開發(fā)源碼深入分析

    SpringBoot web開發(fā)源碼深入分析

    Web開發(fā)的核心內(nèi)容主要包括內(nèi)嵌的Servlet容器和SpringMVCSpringBoot使用起來(lái)非常簡(jiǎn)潔,大部分配置都有SpringBoot自動(dòng)裝配,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧
    2022-10-10
  • Mybatis注解方式操作Oracle數(shù)據(jù)庫(kù)詳解

    Mybatis注解方式操作Oracle數(shù)據(jù)庫(kù)詳解

    這篇文章主要介紹了Mybatis注解方式操作Oracle數(shù)據(jù)庫(kù)詳解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2020-11-11
  • 基于Java 生產(chǎn)者消費(fèi)者模式(詳細(xì)分析)

    基于Java 生產(chǎn)者消費(fèi)者模式(詳細(xì)分析)

    下面小編就為大家分享一篇基于Java 生產(chǎn)者消費(fèi)者模式(詳細(xì)分析),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧
    2018-01-01
  • java實(shí)現(xiàn)文件重命名

    java實(shí)現(xiàn)文件重命名

    這篇文章主要為大家詳細(xì)介紹了java實(shí)現(xiàn)文件重命名,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-03-03
  • Java利用iTextPDF庫(kù)實(shí)現(xiàn)制作PDF表格模板并填充數(shù)據(jù)

    Java利用iTextPDF庫(kù)實(shí)現(xiàn)制作PDF表格模板并填充數(shù)據(jù)

    這篇文章主要為大家詳細(xì)介紹了如何通過(guò)Java的iTextPDF庫(kù)制作一個(gè)PDF表格模板并填充數(shù)據(jù),文中的示例代碼講解詳細(xì),感興趣的小伙伴快跟隨小編一起學(xué)習(xí)一下吧
    2023-12-12

最新評(píng)論