高并发场景下的热点key问题探析与应对策略

news2025/2/24 5:54:19

目录

一、问题描述

二、发现机制

三、解决策略分析

 (一)解决策略一:多级缓存策略

客户端本地缓存

代理节点本地缓存

 (二)解决策略二:多副本策略

 (三)解决策略三:热点 Key 拆分与动态分散

四、总结


干货分享,感谢您的阅读!

在高并发场景下,缓存作为前置查询机制,显著减轻了数据库的压力,提高了系统性能。然而,这也带来了缓存失效、增加回溯率等风险。常见的问题包括缓存穿透、缓存雪崩、热Key和大Key等。这些问题如果不加以处理,会影响系统的稳定性和性能。因此,采用有效的缓存策略,如缓存空结果、布隆过滤器、缓存过期时间随机化、多级缓存等,对于保障系统在高并发情况下的可靠性至关重要。本次我们将详细探讨热点key及其应对策略。

历史缓存热门问题回顾:

热门问题具体分析和解决方案
缓存穿透高并发场景下的缓存穿透问题探析与应对策略-CSDN博客
缓存雪崩高并发场景下的缓存雪崩探析与应对策略-CSDN博客
缓存击穿高并发场景下的缓存击穿问题探析与应对策略-CSDN博客
大 Key问题高并发场景下的大 Key 问题及应对策略-CSDN博客
热点Key发现机制优化分布式系统性能:热key识别与实战解决方案-CSDN博客

一、问题描述

热点 key 问题是指某些数据的访问量非常高,超过了缓存服务器的处理能力。这种现象在电商促销、社交媒体热点等场景中特别常见。热点 key 问题主要有以下几个方面:

  1. 流量集中,达到物理网卡上限:当大量请求集中到某个热点 key 时,这些请求会被路由到相同的缓存服务器。随着流量增加,服务器的物理网卡可能达到带宽上限,无法再处理更多请求。
  2. 请求过多,缓存分片服务被打垮:缓存系统通常使用分片机制来分担负载。然而,热点 key 的访问量可能过高,单个分片无法处理,导致该分片服务被打垮。
  3. 缓存分片打垮,重建再次被打垮,引起业务雪崩:当某个缓存分片被打垮后,系统可能会尝试重建该分片。然而,重建过程中的负载再次集中到该分片上,导致分片再次被打垮,形成恶性循环,引起业务系统的雪崩。

二、发现机制

本部分可直接见:优化分布式系统性能:热key识别与实战解决方案-CSDN博客

在现代分布式系统中,热key问题已经成为影响系统性能和稳定性的重要因素之一。热key,指的是在分布式缓存系统中某些特定的key被频繁访问,导致这些key所在节点的负载过高,甚至可能导致系统瓶颈或崩溃。尽管我们可以通过本地缓存、热key备份和迁移等方式来解决热key问题,但如果热key已经出现而没有及时发现和处理,问题将变得更加棘手。因此,如何提前发现并及时处理热key,是保障系统稳定性和性能的关键。

通过人为预测,客户端监控,机器层面监控,Redis服务端Monitor以及热点发现系统等多种手段,可以及时识别并处理潜在的热点key。每种解决方案都有其独特的优势和局限性,应根据具体业务场景选择合适的策略进行实施。

在实施过程中,需要关注解决方案的实时性、成本效益以及对现有系统的影响。同时,建议采用综合的监控和预测机制,持续优化和调整策略,以确保系统在面对高并发和复杂业务场景时能够稳定可靠地运行。热key问题的解决不仅是技术层面的挑战,更是对系统架构设计和运维管理能力的综合考验。通过有效的热key管理,可以提升系统的响应速度和整体性能,为用户提供更加稳定和高效的服务体验。

三、解决策略分析

 (一)解决策略一:多级缓存策略

多级缓存策略通过在客户端和服务端都设置缓存层,以便将缓存离用户更近,从而减少对远程缓存服务器的访问。

客户端本地缓存

在客户端加入本地缓存,如使用 Guava Cache 或 Ehcache,热点数据可以直接命中本地缓存,从根本上减少热点请求到缓存服务的次数。

  • 优点:减少网络延迟,提高缓存命中率,降低远程缓存服务器压力。
  • 缺点:容量有限,容易受到业务数据的入侵。

可以通过改造 Redis SDK 集成本地缓存功能,从而对业务代码无感知:

import com.google.common.cache.CacheBuilder;
import com.google.common.cache.CacheLoader;
import com.google.common.cache.LoadingCache;

import java.util.concurrent.ExecutionException;
import java.util.concurrent.TimeUnit;

/**
 * @program: zyfboot-javabasic
 * @author: zhangyanfeng
 * @create: 2013-03-23 22:33
 **/
public class LocalCache {

    private static final LoadingCache<String, String> localCache = CacheBuilder.newBuilder()
            .maximumSize(1000)
            .expireAfterWrite(10, TimeUnit.MINUTES)
            .build(new CacheLoader<String, String>() {
                @Override
                public String load(String key) throws Exception {
                    // 默认返回空值,可以改为从远程缓存或数据库加载数据
                    return null;
                }
            });

    public static String get(String key) {
        try {
            return localCache.get(key);
        } catch (ExecutionException e) {
            e.printStackTrace();
            return null;
        }
    }

    public static void put(String key, String value) {
        localCache.put(key, value);
    }

    public static void main(String[] args) {
        // 示例:设置和获取本地缓存
        LocalCache.put("hot_key", "hot_value");
        System.out.println("Local cache value: " + LocalCache.get("hot_key"));
    }
}

代理节点本地缓存

如果缓存集群为代理模式,可以在代理节点上添加本地缓存。代理节点可以水平扩展,通过分散压力解决容量有限的问题。

  • 优点:缓存容量可以扩展,通过代理节点减少远程缓存服务器的压力。
  • 缺点:性能稍逊于客户端本地缓存,因为代理节点距离用户较远。

 (二)解决策略二:多副本策略

多副本策略的基本思路是为热点 key 创建多个副本,并将这些副本分布在不同的缓存节点上。客户端在读取数据时,可以随机选择一个副本节点进行读取,从而分散读取请求,减轻单个节点的压力。多副本策略的实现需要解决以下几个问题:

  1. 副本创建和同步:需要确保热点 key 的多个副本在创建后能够及时同步更新,以保证数据一致性。
  2. 读取请求分发:客户端在读取数据时,需要能够随机选择一个副本节点进行读取。
  3. 一致性保证:需要处理多副本之间的数据一致性问题,尤其是在写操作较多的场景下。

以下是一个简单的多副本策略实现示例,基于 Redis 的主从复制机制:

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

import java.util.ArrayList;
import java.util.List;
import java.util.Random;

/**
 * @program: zyfboot-javabasic
 * @author: zhangyanfeng
 * @create: 2013-03-24 21:12
 **/
public class MultiReplicaCache {

    private static final int NUM_REPLICAS = 3;
    private static final List<JedisPool> replicaPools = new ArrayList<>();
    private static final Random random = new Random();

    static {
        for (int i = 0; i < NUM_REPLICAS; i++) {
            JedisPool pool = new JedisPool(new JedisPoolConfig(), "localhost", 6379 + i);
            replicaPools.add(pool);
        }
    }

    public static void set(String key, String value) {
        try (Jedis jedis = replicaPools.get(0).getResource()) {
            jedis.set(key, value);
        }
        for (int i = 1; i < NUM_REPLICAS; i++) {
            try (Jedis jedis = replicaPools.get(i).getResource()) {
                jedis.slaveof("localhost", 6379);
            }
        }
    }

    public static String get(String key) {
        int replicaIndex = random.nextInt(NUM_REPLICAS);
        try (Jedis jedis = replicaPools.get(replicaIndex).getResource()) {
            return jedis.get(key);
        }
    }

    public static void main(String[] args) {
        String key = "hot_key";
        String value = "hot_value";
        MultiReplicaCache.set(key, value);
        System.out.println("Cache value: " + MultiReplicaCache.get(key));
    }
}

可以看到直接的优点是:1.分散读取压力:多个副本可以显著分散读取请求,减少单个节点的压力;2.提高读取性能:通过多副本并行读取,提高系统的整体读取性能。

但重点需要关注其存在的两大基本问题

  • 一致性问题:多副本之间的数据同步可能会导致一致性问题,特别是在写操作频繁的情况下。
  • 资源消耗增加:创建多个副本会增加存储和网络资源的消耗。

 (三)解决策略三:热点 Key 拆分与动态分散

动态分散热点 key 的基本思路是在存储热点 key 时,将其拆分成多个子 key,并将这些子 key 分布到不同的分片上进行存储。在读取数据时,通过组合子 key 的结果来还原原始数据。这种方法可以显著分散对单个热点 key 的访问压力。

实现热点 key 动态分散思路:

  1. 拆分热点 Key:将一个热点 key 拆分成多个子 key。
  2. 分布式存储子 Key:将子 key 分布到不同的分片上进行存储。
  3. 组合读取子 Key:在读取数据时,通过组合子 key 的结果来还原原始数据。

简单实现如下:

import redis.clients.jedis.Jedis;
import redis.clients.jedis.JedisPool;
import redis.clients.jedis.JedisPoolConfig;

import java.util.ArrayList;
import java.util.List;
import java.util.Random;

/**
 * @program: zyfboot-javabasic
 * @author: zhangyanfeng
 * @create: 2013-03-24 21:35
 **/
public class HotKeyDistribution {

    private static final int NUM_PARTS = 5;
    private static final List<JedisPool> shardPools = new ArrayList<>();
    private static final Random random = new Random();

    static {
        for (int i = 0; i < NUM_PARTS; i++) {
            JedisPool pool = new JedisPool(new JedisPoolConfig(), 
                                           "localhost", 6379 + i);
            shardPools.add(pool);
        }
    }

    public static void set(String key, String value) {
        int partLength = value.length() / NUM_PARTS;
        for (int i = 0; i < NUM_PARTS; i++) {
            String partKey = key + "_" + i;
            String partValue = value.substring(i * partLength, 
                                (i + 1) * partLength);
            try (Jedis jedis = shardPools.get(i % shardPools.size()).getResource()) {
                jedis.set(partKey, partValue);
            }
        }
    }

    public static String get(String key) {
        StringBuilder value = new StringBuilder();
        for (int i = 0; i < NUM_PARTS; i++) {
            String partKey = key + "_" + i;
            try (Jedis jedis = shardPools.get(i % shardPools.size()).getResource()) {
                value.append(jedis.get(partKey));
            }
        }
        return value.toString();
    }

    public static void main(String[] args) {
        String key = "hot_key";
        String value = "this_is_a_very_hot_key_value_with_large_size";
        HotKeyDistribution.set(key, value);
        System.out.println("Cache value: " + HotKeyDistribution.get(key));
    }
}

四、总结

高并发场景下的热点 key 问题是分布式系统中常见的挑战之一,直接影响系统的性能和稳定性。为了有效应对这一问题,可以采用多级缓存策略、多副本策略以及热点 Key 的拆分与动态分散等多种策略。在实施过程中,需要综合考虑系统的实时性需求、成本效益和对现有架构的影响,持续优化和调整策略,以确保系统在面对复杂的业务场景时能够稳定可靠地运行,为用户提供高效的服务体验。

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

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

相关文章

商标字体的选择:企业和个人申请注册商标攻略!

对于汉字商标&#xff0c;就会涉及到字体的选择&#xff0c;普推商标老杨也经常看到企业因为文字商标字体侵权收到相关字体公司的律师函&#xff0c;所以商标字体选择上要特别注意。 建议选择可以商用的免费字体&#xff0c;常见的有黑体、宋体等&#xff0c;如果这些字体前面…

vue 组件批量删除

element ui table表格中前面这个勾选框 对应 type"selection" 属性&#xff0c;绑定事件时selection-change,当你勾选全面的框时就会触发selection-change 对应的事件 绑定事件里面这样定义方法时&#xff0c;这个val 就是选中的时候那一行的数据&#xff0c;如下图…

数学建模(1):期末大乱炖

1 概述&#xff01;&#xff01; 1.1 原型和模型 原型&#xff1a;客观存在的研究对象称为原型&#xff0c;也称为“系统”、“过程”。 机械系统、电力系统、化学反应过程、生产销售过程等都是原型&#xff1b; 研究原型的结构和原理&#xff0c; 从而进行优化、预测、评价…

02.Linux下安装FFmpeg

目录 一、下载FFmpeg的编译源码 二、编译源码 三、ffmpeg工具结构解析 1、bin目录 2、include库 3、lib库 四、注意事项 五、可能出现的一些问题 1、某些工具未安装/版本过久 2、缺少pkg-config工具 3、缺少ffmplay FFmpeg 是一个开源的跨平台音视频处理工具集&…

QListView自定义item(结合QSqlQueryModel)

QListView:绘制自定义List&#xff08;一&#xff09;——设置ItemDelegate_qt_繁星执着-开放原子开发者工作坊 (csdn.net) QListView自定义Item_qlistview 自定义item-CSDN博客 结合我写的上一篇文章&#xff1a; QTableView与QSqlQueryModel的简单使用-CSDN博客 这次尝试…

[数据集][目标检测]桥梁检测数据集VOC+YOLO格式1116张1类别

数据集格式&#xff1a;Pascal VOC格式YOLO格式(不包含分割路径的txt文件&#xff0c;仅仅包含jpg图片以及对应的VOC格式xml文件和yolo格式txt文件) 图片数量(jpg文件个数)&#xff1a;1116 标注数量(xml文件个数)&#xff1a;1116 标注数量(txt文件个数)&#xff1a;1116 标注…

zfile文件共享系统使用

1.简介 zfile是一款基于java开源的文件共享软件&#xff0c;有两种部署&#xff08;jar包和war包&#xff09;,非常适合广大的Java开发人员使用&#xff08;当然其他人员也可使用&#xff09;&#xff0c;或者你也可以称它为网盘&#xff0c;可以在Windows、MacOS、Linux上部署…

C#串口通信Seriaport和页面传值

串口通信 串口COM&#xff1a;是一种用于连接计算机和外设设备的接口&#xff0c;也叫串行接口&#xff0c;简称com,常见的串口有一半电脑应用的RS-232&#xff08;使用25针或9针的 连接器&#xff09;通俗来讲串口就是usb接口、鼠标串口。键盘串口 串口通讯&#xff1a;是指外…

【MySQL】Linux下MySQL的目录结构、用户、权限与角色

一、Linux下MySQL的目录结构 1、MySQL相关目录 数据库文件存放路径&#xff1a;/var/lib/mysql数据库命令存放路径&#xff1a;/user/bin和/user/sbin配置文件目录&#xff1a;/usr/share/mysql-8.0/、/usr/share/mysql/和/etc/my.cnf 2、假设我们创建了一个数据库dbtest1&a…

uniapp 安卓、IOS、H5、微信小程序实现PDF在线预览

在使用uniapp开发移动端时&#xff0c;微信开发者工具里webview能正常打开后端接口返回的pdf文件流。正式发布后&#xff0c;在配置了业务域名和服务器域名的前提下&#xff0c;预览pdf文件却只能看到白屏&#xff0c;因此我猜测微信小程序不能通过webview读取文件流。这个想法…

论文阅读_基于嵌入的Facebook搜索

英文名称&#xff1a;Embedding-based Retrieval in Facebook Search 中文名称&#xff1a;基于嵌入式检索的Facebook搜索 时间&#xff1a;Wed, 29 Jul 2020 (v2) 地址&#xff1a;https://arxiv.org/abs/2006.11632 作者&#xff1a;Jui-Ting Huang, Ashish Sharma, Shuying …

09_计算机网络模型

目录 OSI/RM七层模型 OSI/RM七层模型 各层介绍及硬件设备 传输介质 TCP/IP协议簇 网络层协议 传输层协议 应用层协议 完整URL的组成 IP地址表示与计算 分类地址格式 子网划分和超网聚合 无分类编址 特殊含义的IP地址 IPv6协议 过渡技术 OSI/RM七层模型 OSI/RM七…

《昇思25天学习打卡营第6天 | 函数式自动微分》

《昇思25天学习打卡营第6天 | 函数式自动微分》 目录 《昇思25天学习打卡营第6天 | 函数式自动微分》函数式自动微分简单的单层线性变换模型函数与计算图微分函数与梯度计算Stop Gradient 函数式自动微分 神经网络的训练主要使用反向传播算法&#xff0c;模型预测值&#xff0…

LeetCode题练习与总结:重排链表--143

一、题目描述 给定一个单链表 L 的头节点 head &#xff0c;单链表 L 表示为&#xff1a; L0 → L1 → … → Ln - 1 → Ln请将其重新排列后变为&#xff1a; L0 → Ln → L1 → Ln - 1 → L2 → Ln - 2 → … 不能只是单纯的改变节点内部的值&#xff0c;而是需要实际的进…

centos 7.9 离线环境安装GPU服务环境

文章目录 centos 7.9 离线环境安装GPU服务环境系统配置更新 gcc更新内核安装显卡驱动安装cuda安装docker 和 nvidia-container-runtime验证 centos 7.9 离线环境安装GPU服务环境 基于centos 7.9 离线安装gpu 服务基础环境&#xff0c;用于在docker 中运行算法服务 系统配置 …

python open函数中文乱码怎么解决

首先在D盘下新建一个html文档&#xff0c;接着在里面输入含有中文的Html字符&#xff0c;使用中文格式对读取的字符进行解码&#xff0c;再用utf-8的模式对字符进行编码&#xff0c;然后就能正确输出中文字符。 代码如下&#xff1a; # -*- coding: UTF-8 -*- file1 open(&quo…

Python | Leetcode Python题解之第207题课程表

题目&#xff1a; 题解&#xff1a; class Solution:def canFinish(self, numCourses: int, prerequisites: List[List[int]]) -> bool:edges collections.defaultdict(list)indeg [0] * numCoursesfor info in prerequisites:edges[info[1]].append(info[0])indeg[info[…

【Web3项目案例】Ethers.js极简入门+实战案例:实现ERC20协议代币查询、交易

苏泽 大家好 这里是苏泽 一个钟爱区块链技术的后端开发者 本篇专栏 ←持续记录本人自学智能合约学习笔记和经验总结 如果喜欢拜托三连支持~ 目录 简介 前景科普-ERC20 Ethers极简入门教程&#xff1a;HelloVitalik&#xff08;非小白可跳&#xff09; 教程概览 开发工具 V…

LeetCode-刷题记录-二分法合集(本篇blog会持续更新哦~)

一、二分查找概述 二分查找&#xff08;Binary Search&#xff09;是一种高效的查找算法&#xff0c;适用于有序数组或列表。&#xff08;但其实只要满足二段性&#xff0c;就可以使用二分法&#xff0c;本篇博客后面博主会持续更新一些题&#xff0c;来破除一下人们对“只有有…

44 - 50题高级字符串函数 / 正则表达式 / 子句 - 高频 SQL 50 题基础版

目录 1. 相关知识点2.例子2.44 - 修复表中的名字2.45 - 患某种疾病的患者2.46 - 删除重复的电子邮箱2.47 - 第二高的薪水2.48 - 按日期分组销售产品2.49 - 列出指定时间段内所有的下单产品2.50 - 查找拥有有效邮箱的用户 1. 相关知识点 相关函数 函数含义concat()字符串拼接upp…