Redis分布式缓存面试题

为什么使用分布式缓存?

1. 提升性能

  • 降低延迟:将数据缓存在离应用更近的地方,减少数据访问时间。
  • 减轻数据库压力:缓存频繁访问的数据,减少对后端数据库的请求,提升系统响应速度。

2. 扩展性

  • 水平扩展:通过增加节点,分布式缓存可以轻松扩展,处理更大规模的数据和请求。
  • 负载均衡:数据分布在不同节点上,避免单点瓶颈,提升系统整体吞吐量。

3. 高可用性

  • 容错能力:即使某个节点故障,其他节点仍能继续提供服务,确保系统稳定运行。
  • 数据冗余:通过数据复制,防止单点故障导致的数据丢失。

4. 支持高并发

  • 应对大量请求:分布式缓存能有效处理高并发场景,确保系统在高负载下仍能快速响应。

为什么使用Redis做分布式缓存?

1. 高性能

  • 内存存储,读写速度快。
  • 单线程模型,避免竞争问题,支持高并发。

2. 丰富的数据结构

  • 支持字符串、哈希、列表、集合、有序集合等。

3. 持久化支持

  • RDB 快照和 AOF 日志,确保数据不丢失。

4. 高可用性

  • 主从复制、哨兵模式、集群模式。

5. 分布式支持

  • Redis Cluster 支持数据分片和动态扩展。

6. 丰富的功能

  • Lua 脚本、过期机制、发布/订阅、事务。

面对缓存穿透问题,有什么解决办法?

1. 缓存空值

  • 将空结果缓存,设置较短过期时间。

2. 布隆过滤器

  • 快速判断数据是否存在,过滤无效请求。

3. 缓存预热

  • 提前加载热点数据到缓存。

4. 限流和降级

  • 限制请求量或返回默认值。

数据库更新时布隆过滤器的同步方案

1. 定期重新建布隆过滤器

  • 定期(每天或每小时)重新加载数据库中的有效键构建布隆过滤器。

2. 使用计数布隆过滤器

  • 通过对每个key进行计数,支持动态删除和更新。

3. 结合缓存

  • 通过缓存和布隆过滤器的组合实现实时更新。

4. 使用布隆过滤器的变种

  • 如 Scalable Bloom Filter,适合动态数据量。

介绍一下分层布隆过滤器Scalable Bloom Filter

Scalable Bloom Filter 是布隆过滤器的一种变体,旨在解决传统布隆过滤器在数据量动态增长时的局限性。传统布隆过滤器需要预先设定容量,如果实际数据量超过预设容量,误判率会显著增加。而 Scalable Bloom Filter 可以动态扩展,适应数据量的增长。


Scalable Bloom Filter 的核心思想

  1. 分层设计

    • Scalable Bloom Filter 由多个布隆过滤器层(Layer)组成。
    • 每一层都是一个独立的布隆过滤器,容量和误判率可以单独设置。
    • 当某一层的容量接近饱和时,会自动创建新的层。
  2. 动态扩展

    • 当数据量增加时,新的数据会被添加到最新的层中。
    • 查询时,会依次检查每一层,直到找到匹配的层或确认数据不存在。
  3. 误判率控制

    • 每一层的误判率可以单独设置,通常随着层数的增加,误判率逐渐降低。
    • 整体误判率是所有层误判率的累积结果。

Scalable Bloom Filter 的优点

  1. 动态扩容:无需预先设定容量,适合数据量动态增长的场景。
  2. 误判率可控:通过分层设计,可以有效控制整体误判率。
  3. 灵活性高:可以根据需求调整每一层的容量和误判率。

Scalable Bloom Filter 的缺点

  1. 内存占用较高:由于分层设计,每一层都需要独立的内存空间。
  2. 查询性能稍低:查询时需要依次检查每一层,性能略低于单层布隆过滤器。
  3. 实现复杂度较高:需要管理多个布隆过滤器层。

Java 实现

以下是 Scalable Bloom Filter 的简单实现:

import com.google.common.hash.BloomFilter;
import com.google.common.hash.Funnels;
import java.util.ArrayList;
import java.util.List;public class ScalableBloomFilter {private List<BloomFilter<String>> filters; // 布隆过滤器层private int layerCapacity; // 每一层的容量private double falsePositiveRate; // 每一层的误判率public ScalableBloomFilter(int layerCapacity, double falsePositiveRate) {this.filters = new ArrayList<>();this.layerCapacity = layerCapacity;this.falsePositiveRate = falsePositiveRate;addLayer(); // 初始化第一层}/*** 添加一个新层*/private void addLayer() {BloomFilter<String> newLayer = BloomFilter.create(Funnels.stringFunnel(), layerCapacity, falsePositiveRate);filters.add(newLayer);}/*** 添加一个元素*/public void add(String value) {// 如果当前层已满,添加新层if (filters.get(filters.size() - 1).approximateElementCount() >= layerCapacity) {addLayer();}// 将元素添加到最新的层filters.get(filters.size() - 1).put(value);}/*** 检查元素是否存在*/public boolean mightContain(String value) {// 依次检查每一层for (BloomFilter<String> filter : filters) {if (filter.mightContain(value)) {return true;}}return false;}/*** 获取当前层数*/public int getLayerCount() {return filters.size();}
}

使用示例

public class ScalableBloomFilterExample {public static void main(String[] args) {ScalableBloomFilter scalableBloomFilter = new ScalableBloomFilter(1000, 0.01);// 添加元素scalableBloomFilter.add("key1");scalableBloomFilter.add("key2");// 检查元素是否存在System.out.println("Contains key1: " + scalableBloomFilter.mightContain("key1")); // trueSystem.out.println("Contains key3: " + scalableBloomFilter.mightContain("key3")); // false// 获取当前层数System.out.println("Layer count: " + scalableBloomFilter.getLayerCount()); // 1}
}

Scalable Bloom Filter 的应用场景

  1. 动态数据量场景:如实时日志处理、用户行为分析等。
  2. 分布式系统:如分布式缓存、分布式数据库的去重。
  3. 大数据处理:如海量数据的快速过滤和查询。

总结

Scalable Bloom Filter 通过分层设计和动态扩展,解决了传统布隆过滤器在数据量动态增长时的局限性。它的核心优势在于:

  1. 动态扩容:无需预先设定容量。
  2. 误判率可控:通过分层设计控制整体误判率。
  3. 灵活性高:适合数据量动态变化的场景。

Redis分布式缓存如何判断热点数据?

1. 基于访问频率

  • 原理:通过统计每个键的访问频率(如每秒访问次数),识别出访问频率最高的数据。
  • 实现方法
    • 使用 Redis 的 INCR 命令或监控工具(如 Redis Monitor)统计键的访问频率。
    • 使用 Lua 脚本或客户端代码记录每个键的访问次数。

Java 实现

import redis.clients.jedis.Jedis;public class HotKeyDetector {private Jedis jedis;public HotKeyDetector(Jedis jedis) {this.jedis = jedis;}public void trackAccess(String key) {// 使用 Redis 的计数器记录每个键的访问次数jedis.incr("access_count:" + key);}public String getMostFrequentKey() {// 获取所有键的访问计数Set<String> keys = jedis.keys("access_count:*");String hotKey = null;long maxCount = 0;for (String key : keys) {long count = Long.parseLong(jedis.get(key));if (count > maxCount) {maxCount = count;hotKey = key.replace("access_count:", "");}}return hotKey;}
}

2. 基于时间窗口

  • 原理:在特定的时间窗口内(如最近 1 分钟)统计键的访问频率,识别出热点数据。
  • 实现方法
    • 使用 Redis 的 ZSET(有序集合)记录每个键的访问时间戳。
    • 定期清理过期的访问记录,并统计时间窗口内的访问次数。

Java 实现

import redis.clients.jedis.Jedis;public class TimeWindowHotKeyDetector {private Jedis jedis;private static final long WINDOW_SIZE = 60000; // 时间窗口大小(1 分钟)public TimeWindowHotKeyDetector(Jedis jedis) {this.jedis = jedis;}public void trackAccess(String key) {long currentTime = System.currentTimeMillis();// 使用 ZSET 记录访问时间戳jedis.zadd("access_times:" + key, currentTime, String.valueOf(currentTime));// 清理时间窗口之外的数据jedis.zremrangeByScore("access_times:" + key, 0, currentTime - WINDOW_SIZE);}public String getMostFrequentKey() {Set<String> keys = jedis.keys("access_times:*");String hotKey = null;long maxCount = 0;for (String key : keys) {long count = jedis.zcard(key);if (count > maxCount) {maxCount = count;hotKey = key.replace("access_times:", "");}}return hotKey;}
}

3. 基于采样统计

  • 原理:通过采样部分请求,统计键的访问频率,推断出热点数据。
  • 实现方法
    • 使用 Redis 的 MONITOR 命令或客户端代码采样请求。
    • 对采样数据进行分析,识别出高频访问的键。

4. 使用 Redis 模块(如 RedisGears)

  • 原理:利用 RedisGears 这样的扩展模块,实时监控和分析键的访问模式。
  • 实现方法
    • 编写 RedisGears 脚本,统计键的访问频率并输出热点数据。

5. 基于外部监控工具

  • 原理:使用外部监控工具(如 Prometheus、Grafana)收集 Redis 的访问数据,并通过可视化或分析工具识别热点数据。
  • 实现方法
    • 配置 Redis 的监控插件,将访问数据导出到监控工具。
    • 在监控工具中设置告警规则或分析报告。

总结

判断 Redis 分布式缓存中的热点数据可以通过以下方法:

  1. 基于访问频率:统计每个键的访问次数。
  2. 基于时间窗口:统计特定时间窗口内的访问频率。
  3. 基于采样统计:通过采样请求推断热点数据。
  4. 使用 Redis 模块:如 RedisGears 实时监控。
  5. 基于外部监控工具:如 Prometheus、Grafana。

明日继续更新 😊

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

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

相关文章

基于阿里云PAI平台快速部署DeepSeek大模型实战指南

一、DeepSeek大模型&#xff1a;企业级AI应用的新标杆 1.1 为什么选择DeepSeek&#xff1f; 近期&#xff0c;DeepSeek系列模型凭借其接近GPT-4的性能和开源策略&#xff0c;成为全球开发者关注的焦点。在多项国际评测中&#xff0c;DeepSeek-R1模型在推理能力、多语言支持和…

C++---了解STL

上节学习了模板&#xff0c;那么就得谈到C的标准模板库STL。 C98&#xff1a;以模板方式重写了C标准库&#xff0c;引入了STL(标准模板库)。 1.概念 STL(Standard template Libarary)标准模板库&#xff1a;是C标准库的重要组成部分&#xff0c;不仅是一个可复用的组件库&am…

分享几款比较常用的接口测试工具

首先&#xff0c;什么是接口呢&#xff1f; 接口一般来说有两种&#xff0c;一种是程序内部的接口&#xff0c;一种是系统对外的接口。 系统对外的接口&#xff1a;比如你要从别的网站或服务器上获取资源或信息&#xff0c;别人肯定不会把数据库共享给你&#xff0c;他只能给你…

Qt layout

文章目录 Qt layout**关键机制****验证示例****常见误区****最佳实践****总结**关键点总结&#xff1a;示例代码说明&#xff1a;结论&#xff1a; Qt layout 在 Qt 中&#xff0c;当调用 widget->setLayout(layout) 时&#xff0c;layout 的父对象会被自动设置为该 widget…

flutter: table calendar笔记

pub dev&#xff1a;table_calendar 3.2.0 我来详细解释 TableCalendar 是如何根据不同的 CalendarFormat 来显示界面的。主要逻辑在 CalendarCore 中实现。 核心逻辑分为以下几个部分&#xff1a; 页面数量计算 - _getPageCount 方法根据不同格式计算总页数&#xff1a; in…

【C++】各个版本新的特性和改进

C 语言自从其诞生以来&#xff0c;经历了多个版本的更新&#xff0c;每个版本都引入了新的特性和改进&#xff0c;目的是提升语言的表达能力、性能、安全性以及开发效率。下面是各个主要版本&#xff08;从 C98 到 C20&#xff09;的一些关键特性。 C98 (1998年) ISO C 标准化…

C++模板与STL七日斩:从工业编程到高效数据管理(工业项目)

模板如何提升工业代码复用性 实战项目&#xff1a;创建通用【工业设备容器】模板类 类模板的定义与实例化模板参数默认值 #include <iostream> #include <string> using namespace std;template <typename T string> class IndustrialContainer { priva…

sh脚本把服务器B,服务器C目录的文件下载到服务器A目录,添加开机自启动并且一小时执行一次脚本

脚本逻辑 第一次会下载,第二次比较如果有就不下载 文件已存在&#xff1a; 如果目标目录中已经存在同名文件&#xff0c;rsync 会比较源文件和目标文件的大小和修改时间。 如果源文件和目标文件的大小和修改时间完全相同&#xff0c;rsync 会跳过该文件&#xff0c;不会重新下载…

云手机如何进行经纬度修改

云手机如何进行经纬度修改 云手机修改经纬度的方法因不同服务商和操作方式有所差异&#xff0c;以下是综合多个来源的常用方法及注意事项&#xff1a; 通过ADB命令注入GPS数据&#xff08;适用于技术用户&#xff09; 1.连接云手机 使用ADB工具连接云手机服务器&#xff0c;…

透彻理解:方差、协方差、相关系数、协方差矩阵及其应用

最近看了几篇跨领域特征对齐方面的经典文献&#xff0c;学者们搞了很多花样&#xff0c;如有的提出一阶统计特征对齐&#xff0c;有的提出二阶统计特征对齐&#xff0c;有的学者提出高阶统计特征对齐。 通俗而言&#xff0c;就是在统计特征层面对跨域特征进行对齐&#xff0c;…

Unity基础学习(二)

二、Mono中的重要内容 1、延迟函数 &#xff08;1&#xff09;延迟函数定义 延迟执行的函数&#xff0c;可以设定要延迟执行的函数和具体延迟的时间 &#xff08;2&#xff09;延迟函数的使用 #region 1、延迟函数//函数&#xff1a;Invoke(函数名/字符串&#xff0c;延迟时…

20250212:ZLKMedia 推流

1:资料 快速开始 ZLMediaKit/ZLMediaKit Wiki GitHub GitHub - ZLMediaKit/ZLMediaKit: WebRTC/RTSP/RTMP/HTTP/HLS/HTTP-FLV/WebSocket-FLV/HTTP-TS/HTTP-fMP4/WebSocket-TS/WebSocket-fMP4/GB28181/SRT server and client framework based on C++11 文档里面提供了各个系…

Holoens2开发报错记录02_通过主机获取彩色和深度数据流常见错误

01.E1696 E1696 无法打开源文件 “stdio.h” 解决方法&#xff1a; 更新一下SDK 1&#xff09;打开Visual Studio Installer&#xff0c;点击修改 2&#xff09;安装详细信息中自己系统对应的SDK&#xff0c;点击修改即可 02.WinError 10060 方法来源 解决方法&#xff1a…

【Qt之QQuickWidget】QML嵌入QWidget中

由于我项目开始使用Widgets,换公司后直接使用QML开发&#xff0c;没有了解过如何实现widget到qml过渡&#xff0c;恰逢面试时遇到一家公司希望从widget迁移到qml开发&#xff0c;询问相关实现&#xff0c;一时语塞&#xff0c;很尴尬&#xff0c;粗略研究并总结下。 对qwidget嵌…

从单片机的启动说起一个单片机到点灯发生了什么下——使用GPIO点一个灯

目录 前言 HAL库对GPIO的抽象 核心分析&#xff1a;HAL_GPIO_Init 前言 我们终于到达了熟悉的地方&#xff0c;对GPIO的初始化。经过漫长的铺垫&#xff0c;我们终于历经千辛万苦&#xff0c;来到了这里。关于GPIO的八种模式等更加详细的细节&#xff0c;由于只是点个灯&am…

ESP32S3:解决RWDT无法触发中断问题,二次开发者怎么才能使用内部RTC看门狗中断RWDT呢?

目录 基于ESP32S3:解决RWDT无法触发中断问题引言解决方案1. 查看报错日志2. 分析报错及一步一步找到解决方法3.小结我的源码基于ESP32S3:解决RWDT无法触发中断问题 引言 在嵌入式系统中,RWDT(看门狗定时器)是确保系统稳定性的重要组件。然而,在某些情况下,RWDT可能无法…

对计算机中缓存的理解和使用Redis作为缓存

使用Redis作为缓存缓存例子缓存的引入 Redis缓存的实现 使用Redis作为缓存 缓存 ​什么是缓存&#xff0c;第一次接触这个东西是在考研学习408的时候&#xff0c;计算机组成原理里面学习到Cache缓存&#xff0c;用于降低由于内存和CPU的速度的差异带来的延迟。它是在CPU和内存…

vue3的实用工具库@vueuse/core

1.什么是vueuse/core 是一个基于 ‌Vue Composition API‌ 开发的实用工具库&#xff0c;旨在通过封装高频功能为可复用的组合式函数&#xff08;Composables&#xff09;&#xff0c;简化 Vue 应用的开发流程。 提供 ‌200 开箱即用的函数‌&#xff0c;覆盖状态管理、浏览器…

基于SSM的《计算机网络》题库管理系统(源码+lw+部署文档+讲解),源码可白嫖!

摘 要 《计算机网络》题库管理系统是一种新颖的考试管理模式&#xff0c;因为系统是用Java技术进行开发。系统分为三个用户进行登录并操作&#xff0c;分别是管理员、教师和学生。教师在系统后台新增试题和试卷&#xff0c;学生进行在线考试&#xff0c;还能对考生记录、错题…

C++初阶——简单实现stack和queue

目录 1、Deque(了解) 1.1 起源 1.2 结构 1.3 优缺点 1.4 应用 2、Stack 3、Queue 4、Priority_Queue 注意&#xff1a;stack&#xff0c;queue&#xff0c;priority_queue是容器适配器(container adaptor) &#xff0c;封装一个容器&#xff0c;按照某种规则使用&#…