使用本地锁syncronized防止缓存击穿

谷粒商城学习笔记p156

缓存击穿:单个key缓存突然失效,这时大量的请求进行访问,导致数据库压力过大。缓存击穿是指缓存中没有但数据库中有的数据(一般是缓存时间到期),这时由于并发用户特别多,同时读缓存没读到数据,又同时去数据库去取数据,引起数据库压力瞬间增大。

解决办法:加锁,锁有本地锁,也有分布式锁。每种锁有其优缺点和适用场景

本地锁的优点是使用相对来说较为简便,缺点是在分布式应用中锁不住资源。下面是具体实现:

 @Overridepublic Map<String,  List<Catelog2Vo>> getCatelogJson() {String catelogJson = redisTemplate.opsForValue().get("catelogJson");//如果缓存中不存在,则去查库if(StringUtils.isBlank(catelogJson)){System.out.println("缓存未命中......查询数据库..........");Map<String, List<Catelog2Vo>> catelogJsonFromDb = getCatelogJsonFromDb();String s = JSON.toJSONString(collect);//将查库获得的数据添加到缓存中redisTemplate.opsForValue().set("catelogJson",s,1, TimeUnit.DAYS);return catelogJsonFromDb;}System.out.println("缓存命中......直接返回..........");Map<String, List<Catelog2Vo>>map=JSON.parseObject(catelogJson, new TypeReference<Map<String, List<Catelog2Vo>>>() {});return map;}

查询数据库的方法:

    public Map<String,  List<Catelog2Vo>> getCatelogJsonFromDb() {//加同步锁,防止缓存击穿synchronized (this){//获得锁以后,需要去查询缓存,缓存中不存在,才去查库String catelogJson = redisTemplate.opsForValue().get("catelogJson");if(StringUtils.isNotBlank(catelogJson)){Map<String, List<Catelog2Vo>>map = JSON.parseObject(catelogJson, new TypeReference<Map<String, List<Catelog2Vo>>>() {});return map;}System.out.println("查询了数据库..........");List<CategoryEntity> selectList = baseMapper.selectList(null);//获得一级分类的集合List<CategoryEntity> level1Categorys = this.getLevel1Categorys();Map<String, List<Catelog2Vo>> collect = level1Categorys.stream().collect(Collectors.toMap(k -> k.getCatId().toString(), v -> {//根据一级分类id查询其下所有的二级分类List<CategoryEntity> level2Categorys = getCategoryEntities(selectList,v.getCatId());//封装成前端需要的数据格式List<Catelog2Vo> catelog2Vos = null;if (!CollectionUtils.isEmpty(level2Categorys)) {catelog2Vos = level2Categorys.stream().map(level2 -> {//根据二级分类id查询名下所有三级分类信息List<CategoryEntity> level3Categorys = getCategoryEntities(selectList,level2.getCatId());Catelog2Vo catelog2Vo = new Catelog2Vo(v.getCatId().toString(), null, level2.getCatId().toString(), level2.getName());if (!CollectionUtils.isEmpty(level3Categorys)) {List<Catelog2Vo.Catelog3Vo> catelog3VoList = level3Categorys.stream().map(level3 -> {Catelog2Vo.Catelog3Vo catelog3Vo = new Catelog2Vo.Catelog3Vo();catelog3Vo.setCatalog2Id(level2.getCatId().toString());catelog3Vo.setId(level3.getCatId().toString());catelog3Vo.setName(level3.getName());return catelog3Vo;}).collect(Collectors.toList());catelog2Vo.setCatalog3List(catelog3VoList);}return catelog2Vo;}).collect(Collectors.toList());}return catelog2Vos;}));return collect;}}

使用上面的方法仍然存在一定的弊端,在实际测试中也发现查询了2次数据库而不是只查1次。原因在于添加数据到缓存和查库不在同一把锁的操作控制内,也行第一个线程查询到数据释放了锁,然后在添加数据到缓存的间隙,第二个线程获得了锁,但数据还没成功完整添加到缓存中,导致再去查询一次数据库。显然这样设计是不合理的,给数据库造成额外的负担。正确的设计是将添加数据到缓存放在查库的同一把锁控制范围内,保证原子性。附改造后的代码:

    @Overridepublic Map<String,  List<Catelog2Vo>> getCatelogJson() {String catelogJson = redisTemplate.opsForValue().get("catelogJson");//如果缓存中不存在,则去查库if(StringUtils.isBlank(catelogJson)){System.out.println("缓存未命中......查询数据库..........");Map<String, List<Catelog2Vo>> catelogJsonFromDb = getCatelogJsonFromDb();return catelogJsonFromDb;}System.out.println("缓存命中......直接返回..........");Map<String, List<Catelog2Vo>>map=JSON.parseObject(catelogJson, new TypeReference<Map<String, List<Catelog2Vo>>>() {});return map;}

查询数据库的方法:

    public Map<String,  List<Catelog2Vo>> getCatelogJsonFromDb() {//加同步锁,防止缓存击穿synchronized (this){//获得锁以后,需要去查询缓存,缓存中不存在,才去查库String catelogJson = redisTemplate.opsForValue().get("catelogJson");if(StringUtils.isNotBlank(catelogJson)){Map<String, List<Catelog2Vo>>map = JSON.parseObject(catelogJson, new TypeReference<Map<String, List<Catelog2Vo>>>() {});return map;}System.out.println("查询了数据库..........");List<CategoryEntity> selectList = baseMapper.selectList(null);//获得一级分类的集合List<CategoryEntity> level1Categorys = this.getLevel1Categorys();Map<String, List<Catelog2Vo>> collect = level1Categorys.stream().collect(Collectors.toMap(k -> k.getCatId().toString(), v -> {//根据一级分类id查询其下所有的二级分类List<CategoryEntity> level2Categorys = getCategoryEntities(selectList,v.getCatId());//封装成前端需要的数据格式List<Catelog2Vo> catelog2Vos = null;if (!CollectionUtils.isEmpty(level2Categorys)) {catelog2Vos = level2Categorys.stream().map(level2 -> {//根据二级分类id查询名下所有三级分类信息List<CategoryEntity> level3Categorys = getCategoryEntities(selectList,level2.getCatId());Catelog2Vo catelog2Vo = new Catelog2Vo(v.getCatId().toString(), null, level2.getCatId().toString(), level2.getName());if (!CollectionUtils.isEmpty(level3Categorys)) {List<Catelog2Vo.Catelog3Vo> catelog3VoList = level3Categorys.stream().map(level3 -> {Catelog2Vo.Catelog3Vo catelog3Vo = new Catelog2Vo.Catelog3Vo();catelog3Vo.setCatalog2Id(level2.getCatId().toString());catelog3Vo.setId(level3.getCatId().toString());catelog3Vo.setName(level3.getName());return catelog3Vo;}).collect(Collectors.toList());catelog2Vo.setCatalog3List(catelog3VoList);}return catelog2Vo;}).collect(Collectors.toList());}return catelog2Vos;}));String s = JSON.toJSONString(collect);//查询到以后立即将数据库查询结果添加到缓存中,防止时间差,导致第2个获得锁的线程继续查库//将查库和添加到redis缓存中要放在同个锁操作中,保证原子性,redisTemplate.opsForValue().set("catelogJson",s,1, TimeUnit.DAYS);return collect;}}

改造后经测试,终于只查询了一次数据库,查询完库,添加数据到redis,再释放锁。这样第二个获得锁的线程就可直接从缓存中读数据而无须再次查库。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/diannao/24696.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

对待谷歌百度等搜索引擎的正确方式

对待百度、谷歌等搜索引擎的方式是&#xff0c;你要站在搜索引擎之上&#xff0c;保持自己的独立思想和意见。 当谷歌宣布他们将会根据一个名为“Alphabet”的新控股公司来进行业务调整时&#xff0c;在科技界引起了一片恐慌之声。 永远不要说这是一个公司一直在做的事情。不…

一文快速搞懂Nginx —— Nginx 详解

一文快速搞懂Nginx 一、nginx 简介二、正向 / 反向代理2.1 正向代理2.2 反向代理三、负载均衡四、动静分离五、web 缓存六、Nginx 安装6.1 windows版本下的安装6.2 Linux版本下的安装七、常用命令八、为什么选择Nginx 一、nginx 简介 Nginx 同 Apache 一样都是一种 Web 服务器。…

使用 Python 的 Tkinter 来创建 GUI 应用程序

Tkinter 是 Python 自带的一个图形用户界面&#xff08;GUI&#xff09;工具包&#xff0c;它提供了丰富的 GUI 组件和工具&#xff0c;可以用于快速开发各种应用程序 安装和导入 Tkinter 首先&#xff0c;需要确保你已经安装了 Python&#xff0c;并且版本号在 3.0 及以上&a…

嵌入式Linux系统编程 — 3.1 Linux系统中的文件类型

目录 1 Linux 系统中的文件类型简介 2 普通文件 2.1 什么是普通文件 2.2 普通文件分类 3 目录文件 4 字符设备文件和块设备文件 4.1 什么是设备文件 4.2 查看设备文件的方法&#xff1a; 5 符号链接文件&#xff08;link&#xff09; 5.1 什么是符号链接文件 5.2 如…

大模型多轮问答的两种方式

前言 大模型的多轮问答难点就是在于如何精确识别用户最新的提问的真实意图&#xff0c;而在常见的使用大模型进行多轮对话方式中&#xff0c;我接触到的只有两种方式&#xff1a; 一种是简单地直接使用 user 和 assistant 两个角色将一问一答的会话内容喂给大模型&#xff0c…

光伏电站绘制软件的基本方法

随着可再生能源的快速发展&#xff0c;光伏电站的建设日益受到重视。为了提高光伏电站设计的效率和准确性&#xff0c;光伏电站绘制软件的应用变得至关重要。本文将介绍光伏电站绘制软件的基本方法&#xff0c;包括绘制屋顶、屋脊、障碍物和参照物&#xff0c;铺设光伏板&#…

FreeRTOS 阻塞式I2C操作异常 I2C_WaitOnFlagUntilTimeout规避

I2C_WaitOnFlagUntilTimeout 是一个在STM32的HAL库中用于等待I2C操作完成的函数。FreeRTOS是一个可以运行在嵌入式系统上的实时操作系统。在FreeRTOS中&#xff0c;如果你想要实现I2C操作的阻塞式等待&#xff0c;可以使用I2C_WaitOnFlagUntilTimeout函数。 但是&#xff0c;F…

typedef和define的区别

在C语言中&#xff0c;typedef和define都是用来创建别名以增强代码的可读性和可维护性&#xff0c;它们在**执行时间、作用域和功能**等方面存在差异。具体如下&#xff1a; 1. **执行时间** - **typedef**&#xff1a;处理于编译阶段&#xff0c;因此具备类型检查的功能Θic-1…

PCL Loess曲线回归拟合(二维)

文章目录 一、简介二、实现代码三、实现效果参考文献一、简介 LOESS(局部加权回归)回归的原理是基于非参数方法,它主要用于描述两个变量之间复杂的、非线性的关系。LOESS方法的核心在于“局部”和“加权”。它会在每个数据点附近选取一个子集(或称为窗口),并利用这个子集…

使用Python编写Ping监测程序

Ping是一种常用的网络诊断工具&#xff0c;它可以测试两台计算机之间的连通性&#xff1b; 如果您需要监测某个IP地址的连通情况&#xff0c;可以使用Python编写一个Ping监测程序&#xff1b; 本文将介绍如何使用Python编写Ping监测程序 首先&#xff0c;需要导入os、sys、t…

spark复习

第一章 1.​大数据特点:4V 2.​大数据计算模式 3.​hadoop生态系统 4.​spark提供了内存计算和基于DAG的任务调度机制&#xff0c;遵循一个软件栈满足不同应用场景的理念。 5.​hadoop中MapReduce计算框架的缺点&#xff0c;对应的spark的优点 第二章 1.​spark生态系统 …

HTML做成一个炫酷跳动爱心的页面

大家好&#xff0c;今天制作制作一个炫酷跳动爱心的页面&#xff01; 先看具体效果&#xff1a; 要创建一个炫酷跳动爱心的HTML页面&#xff0c;你可以使用HTML、CSS和JavaScript的组合。以下是一个简单的示例&#xff0c;它使用CSS动画和JavaScript来实现跳动效果。 首先&…

vue项目中oss网络图片无法显示的问题

问题说明 如果后台给你烦返回了oss地址的图片,也许会出现如下情况 在图片路径无误的情况下,图片无法正常预览和回显 但是在浏览器中打开却没问题 解决方案 就需要在项目的index.html中做出如下配置,便能正常回显 <meta name"referrer" content"no-referr…

将二叉排序树转换成双向链表--c++【做题记录】

【问题描述】 编写程序在不增加结点的情况下&#xff0c;将二叉排序树转换成有序双向链表&#xff08;如下图&#xff09;。 链表创建结束后&#xff0c;按照从前往后的顺序输出链表中结点的内容。 【输入输出】 【输入形式】 第一行输入数字n&#xff0c;第二行输入n个整数…

LNMP与动静态网站介绍

Nginx发展 Nginx nginx http server Nginx是俄罗斯人 Igor Sysoev(伊戈尔.塞索耶夫)开发的一款高性能的HTTP和反向代理服务器。 Nginx以高效的epoll.kqueue,eventport作为网络IO模型&#xff0c;在高并发场景下&#xff0c;Nginx能够轻松支持5w并发连接数的响应&#xff0c;并…

【文献阅读】LORA: LOW-RANK ADAPTATION OF LARGE LANGUAGE MODELS

目录 1. motivation2. overall3. model3.1 low rank parametrized update matrices3.2 applying lora to transformer 4. limitation5. experiment6. 代码7. 补充参考文献 1. motivation 常规的adaptation需要的微调成本过大现有方法的不足&#xff1a; Adapter Layers Introd…

Vue2入门(安装/创建Vue,安装devtools)

1.下载并安装Vue &#xff08;1&#xff09;Vue是一个基于JavaScript&#xff08;JS&#xff09;实现的框架。要使用它就需要先拿到Vue的js文件&#xff0c;可以从官网(https://v2.cn.vuejs.org/)下载vue.js文件 &#xff08;2&#xff09;下载&#xff1a;开发生产版本更小&a…

Centos7 安装配置SFTP

Centos7安装配置SFTP 更新源安装 OpenSSH 服务启动服务设置为开机自启动新建一个用户 (sftpuser为你要设置的用户的用户名)编辑配置文件设置sftp用户的根目录重启SSH服务代码实现 由于最近工作中需要实现动态上传文件到帆软服务器&#xff0c;但是帆软没有提供相关API&#xff…

c语言面试5(希诺麦田)

由多个源文件组成的C程序&#xff0c;经过编辑、预处理、编译、链接等阶段会生成最终的可执行程序。下面哪个阶段可以发现被调用的函数未定义 A预处理 B编译 C链接 D执行 答案&#xff1a; C。 在链接阶段会检查所有符号的定义和引用是否匹配&#xff0c;当发现有被调用的函数未…

【SpringBoot + Vue 尚庭公寓实战】房间支付方式管理接口实现(三)

【SpringBoot Vue 尚庭公寓实战】房间支付方式管理接口实现&#xff08;三&#xff09; 文章目录 【SpringBoot Vue 尚庭公寓实战】房间支付方式管理接口实现&#xff08;三&#xff09;1、查询全部支付方式列表2、保存或更新支付方式3、根据ID删除支付方式 房间支付方式管理…