springboot中redis的缓存穿透问题实现

什么是缓存穿透问题??

我们使用redis是为了减少数据库的压力,让尽量多的请求去承压能力比较大的redis,而不是数据库。但是高并发条件下,可能会在redis还没有缓存的时候,大量的请求同时进入,导致一大批的请求直奔数据库,而不会经过redis。使用代码模拟缓存穿透问题如下:

首先是service里面的代码:

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

  //springboot自动初始化,不需要我们进行配置,直接注入到代码中使用
  @Autowired
  private RedisTemplate<Object,Object> redisTemplate;

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

    //设置序列化方式,防止乱码
    redisTemplate.setKeySerializer(new StringRedisSerializer());

    //第一步:查询缓存
    News news= (News) redisTemplate.opsForValue().get("newsKey");
    //判断是否存在缓存
    if(null == news){//查询数据库
        news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
        //
        redisTemplate.opsForValue().set("newsKey",news);

        System.out.println("进入数据库。。。。。。。。");

    }else{
      System.out.println("进入缓存。。。。。。。。。");
    }
    return newsDAO.selectByUserIdAndOffset(userId,offset,limit);

  }
}

然后是使用线程池在Controller里面对请求进行模拟:

@Controller
public class HomeController {
  @Autowired
  UserService userService;

  @Autowired
  NewsService newsService;

  //遇到的坑,如果不加method,页面启动不起来。
  @RequestMapping(value = "/home",method = {RequestMethod.GET, RequestMethod.POST})
  @ResponseBody
  public String index(Model model){
    //这边是可以读出数据来的

    //线程池------缓存穿透问题的复现
    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();
  }
}

结果如图:大量的请求进入数据库,那么如何解决这个问题?

方法一、在方法上加锁:

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

  //springboot自动初始化,不需要我们进行配置,直接注入到代码中使用
  @Autowired
  private RedisTemplate<Object,Object> redisTemplate;

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

    //设置序列化方式,防止乱码
    redisTemplate.setKeySerializer(new StringRedisSerializer());

    //第一步:查询缓存
    News news= (News) redisTemplate.opsForValue().get("newsKey");
    //判断是否存在缓存
    if(null == news){
//查询数据库
        news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
        //
        redisTemplate.opsForValue().set("newsKey",news);

        System.out.println("进入数据库。。。。。。。。");

    }else{
      System.out.println("进入缓存。。。。。。。。。");
    }

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

  }
}

直接在方法上加锁,保证每次只有一个请求可以进入。但是这个方法存在一个缺陷,每次只有一个请求可以进入,请求处理的速度变得相当的慢,不利于系统的实时性。

方法二、使用双重校验锁:

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

  //springboot自动初始化,不需要我们进行配置,直接注入到代码中使用
  @Autowired
  private RedisTemplate<Object,Object> redisTemplate;

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

    //设置序列化方式,防止乱码
    redisTemplate.setKeySerializer(new StringRedisSerializer());

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

      //第二种方式:双重检测锁
      synchronized (this){
        //查询数据库
        news = newsDAO.selectByUserIdAndOffset(userId,offset,limit).get(0);
        //
        redisTemplate.opsForValue().set("newsKey",news);

        System.out.println("进入数据库。。。。。。。。");
      }

    }else{
      System.out.println("进入缓存。。。。。。。。。");
    }

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

  }
}

这个方法比较好,虽然不能保证只有一个请求请求数据库,但是当第一批请求进来,第二批之后的所有请求全部会在缓存取数据。

到此这篇关于springboot中redis的缓存穿透问题实现的文章就介绍到这了,更多相关springboot redis缓存穿透内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • Java模拟并解决缓存穿透问题

    什么叫做缓存穿透 缓存穿透只会发生在高并发的时候,就是当有10000个并发进行查询数据的时候,我们一般都会先去redis里面查询进行数据,但是如果redis里面没有这个数据的时候,那么这10000个并发里面就会有很大一部分并发会一下子都去mysql数据库里面进行查询了 解决缓存穿透 首先我模拟一下缓存穿透 比如下面的代码 Pom.xml代码 <?xml version="1.0" encoding="UTF-8"?> <project xmlns=

  • Redis缓存穿透出现原因及解决方案

    在并发式的项目当中,一定要考虑一个缓存穿透的情况.那么什么是缓存穿透呢?简单的说来,就是当大量请求的key根本不在缓存当中,所以导致了请求直接到了数据库上,根本没有经过缓存这一层.比如一个黑客故意制造我们缓存中不存在的key发送大量的请求,就会导致请求直接落到数据库上. 也就是说,缓存穿透就是:1.缓存层不命中.2,存储层不命中,不将空的结果写回缓存.3,返回空结果给客户端. 一般mysql的默认最大连接数是150左右,当然这个是可以用show variables like '%max_conn

  • redis缓存穿透解决方法

    缓存技术可以用来减轻数据库的压力,提升访问效率.目前在企业项目中对缓存也是越来越重视.但是缓存不是说随随便便加入项目就可以了.将缓存整合到项目中,这才是第一步.而缓存带来的穿透问题,进而导致的雪蹦问题都是我们迫切需要解决的问题.本篇文章将我平时项目中的解决方案分享给大家,以供参考. 一.缓存穿透的原理 缓存的正常使用如图: 如图所示,缓存的使用流程: 1.先从缓存中取数据,如果能取到,则直接返回数据给用户.这样不用访问数据库,减轻数据库的压力. 2.如果缓存中没有数据,就会访问数据库. 这里面就

  • springboot中redis的缓存穿透问题实现

    什么是缓存穿透问题?? 我们使用redis是为了减少数据库的压力,让尽量多的请求去承压能力比较大的redis,而不是数据库.但是高并发条件下,可能会在redis还没有缓存的时候,大量的请求同时进入,导致一大批的请求直奔数据库,而不会经过redis.使用代码模拟缓存穿透问题如下: 首先是service里面的代码: @Service public class NewsService { @Autowired private NewsDAO newsDAO; //springboot自动初始化,不需要

  • springboot中redis正确的使用详解

    redis实现了对数据的缓存,在项目里一些字典数据,会话数据,临时性数据都会向redis来存储,而在springboot里对redis也有支持,一般来说多个线程共同使用一个redis实现是有线程安全的风险的,而每个实现一个线程又太浪费资源,无法控制线程数量是非常危险的,所以就出现了一些redis线程池组件,下面说一下两个主要的组件. jedis 线程池主要是每个实例有自己的线程,线程可以从它建立的池子里获取lettuce lettuce是 apache推出的线程池工具,它的redis实例是可以被

  • SpringBoot自定义Redis实现缓存序列化详解

    目录 1.自定义RedisTemplate 1.1.Redis API默认序列化机制 1.2.自定义RedisTemplate序列化机制 1.3.效果测试 2.自定义RedisCacheManager 2.1.Redis注解默认序列化机制 2.2.自定义RedisCacheManager 刚刚完成了Spring Boot整合Redis进行了数据的缓存管理,但缓存管理的实体类数据使用的是JDK序列化方式,不便于使用可视化管理工具进行查看和管理. 接下来分别针对基于注解的Redis缓存实现和基于AP

  • springboot使用Redis作缓存使用入门教程

    1.依赖与数据库设置 <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-data-redis</artifactId> </dependency> <dependency> <groupId>org.apache.commons</groupId> <artifact

  • 详解springboot中redis的使用和分布式session共享问题

    对于分布式使用Nginx+Tomcat实现负载均衡,最常用的均衡算法有IP_Hash.轮训.根据权重.随机等.不管对于哪一种负载均衡算法,由于Nginx对不同的请求分发到某一个Tomcat,Tomcat在运行的时候分别是不同的容器里,因此会出现session不同步或者丢失的问题. 实际上实现Session共享的方案很多,其中一种常用的就是使用Tomcat.Jetty等服务器提供的Session共享功能,将Session的内容统一存储在一个数据库(如MySQL)或缓存(如Redis)中. 本文旨在

  • 浅谈SpringBoot集成Redis实现缓存处理(Spring AOP实现)

    第一章 需求分析 计划在Team的开源项目里加入Redis实现缓存处理,因为业务功能已经实现了一部分,通过写Redis工具类,然后引用,改动量较大,而且不可以实现解耦合,所以想到了Spring框架的AOP(面向切面编程). 开源项目:https://github.com/u014427391/jeeplatform 第二章 SpringBoot简介 Spring框架作为JavaEE框架领域的一款重要的开源框架,在企业应用开发中有着很重要的作用,同时Spring框架及其子框架很多,所以知识量很广.

  • springboot+mybatis+redis 二级缓存问题实例详解

    前言 什么是mybatis二级缓存? 二级缓存是多个sqlsession共享的,其作用域是mapper的同一个namespace. 即,在不同的sqlsession中,相同的namespace下,相同的sql语句,并且sql模板中参数也相同的,会命中缓存. 第一次执行完毕会将数据库中查询的数据写到缓存,第二次会从缓存中获取数据将不再从数据库查询,从而提高查询效率. Mybatis默认没有开启二级缓存,需要在全局配置(mybatis-config.xml)中开启二级缓存. 本文讲述的是使用Redi

  • SpringBoot 开启Redis缓存及使用方法

    目录 Redis缓存 主要步骤 具体实践 整体目录结构 yml文件里配置Redis集群 设置序列化的Bean 编写业务Controller 关于缓存的其他注解 检验结果 之前不是说过Redis可以当作缓存用嘛 现在我们就配置一下SpringBoot使用Redis的缓存 Redis缓存 为什么用Redis作缓存 用redis做缓存,是因为redis有着很优秀的读写能力,在集群下可以保证数据的高可用 主要步骤 1.pom.xml文件添加依赖 2.yml文件配置redis集群 3.编写RedisCon

  • 在项目中使用redis做缓存的一些思路

    目录 在项目中redis做缓存的一些思路 首先,缓存的对象有三种 本人走过的一些弯路 为什么没用Redis做缓存 使用Table作本地缓存 使用Redis作缓存 让我们来思考一下下面几个问题 那么使用本地缓存的问题是什么呢? 什么时候使用Redis? 在项目中redis做缓存的一些思路 首先,缓存的对象有三种 1.数据库中单条的的数据(以表名跟id作为key永久保存到redis),在有更新的地方都要更新缓存(不适用于需要经常更新的数据): 2.对于一些不分页,不需要实时(需要多表查询)的列表,我

  • 详解springboot整合ehcache实现缓存机制

    EhCache 是一个纯Java的进程内缓存框架,具有快速.精干等特点,是Hibernate中默认的CacheProvider. ehcache提供了多种缓存策略,主要分为内存和磁盘两级,所以无需担心容量问题. spring-boot是一个快速的集成框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程.该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置. 由于spring-boot无需任何样板化的配置文件,所以spring-boot集成一些其他框架时会有略微的

随机推荐