【退役之重学Java】Redis 的过期策略

2024-05-11 08:04

本文主要是介绍【退役之重学Java】Redis 的过期策略,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Redis 的过期策略

一、假设设置一个key 只能存活1h,那么1h之后,Redis 是怎么对这批 key 进行删除的?

回答: 定期删除 + 惰性删除

  1. 所谓定期删除,指的是 Redis 默认每隔100ms 就随机抽取一些设置了过期时间的 key,检查其是否过期,如果过期就删除。注意,这里可不是每隔 100ms 就遍历所有设置过期时间的key ,那样就是已超过性能的灾难。实际上,Redis 是每隔 100ms 随机抽取一些 key 来检查和删除的。
  2. 但问题是,定期删除可能会导致过多的 key 到了时间并没有删除掉。所以就需要惰性删除。就是说,在获取某个 key 的时候, Redis 会检查一下,这个 key 如果设置了过期时间,并且是否过期了,如果过期了,此时就会删除。
  3. 通过上述两种手段结合起来,保证过期的 key 一定回被干掉
  4. 但是,实际上这样还是存在一些问题的,如果定期删除漏掉了很多过期的 key, 然后你也没有及时去查,也就没走惰性删除,此时会怎么样?如果大量过期的 key 堆积在内存里,导致 Redis 内存耗尽了,那么怎么办呢?
  5. 答案是:内存淘汰机制

二、内存淘汰机制

如果 Redis 的内存占用过多的是偶,此时就会进行内存太太,有如下一些策略:
1)noeviction: 当内存不足以容纳新写入的数据时,新写入操作会报错,这个一般没人用吧,实在太恶心了
2)allkeys-lru: 当内存不足以容纳新写入的数据时,在键空间中,一处最近最少使用的key(这个是最常用的)
3)allkeys-random: 当内存不足以容纳新写入的数据时,在键空间中,随机移除某个 key(一般没人用)
4) volatile-lru: 当内存不足以容纳新写入的数据时,在设置了过期时间的键空间中,移除最近最少使用的 key(这个一般不太合适)
5)volatile-random: 当内存不足以容纳新写入的数据时,在设置了过期时间的键空间中,随机移除某个 key
6)volatile-ttl: 当内存不足以容纳新写入的数据时,在设置了过期时间的键空间中,有更早过期时间的 key 先移除

手写一个 LRU 算法

下面是一个简单的手写LRU(Least Recently Used)算法的示例代码,使用双向链表和哈希表实现:

import java.util.HashMap;class LRUCache {// 定义双向链表节点class Node {int key;int value;Node prev;Node next;public Node(int key, int value) {this.key = key;this.value = value;}}private HashMap<Integer, Node> map; // 哈希表用于快速查找缓存数据private Node head; // 虚拟头节点private Node tail; // 虚拟尾节点private int capacity; // 缓存容量public LRUCache(int capacity) {this.capacity = capacity;map = new HashMap<>();head = new Node(-1, -1);tail = new Node(-1, -1);head.next = tail;tail.prev = head;}// 获取缓存数据public int get(int key) {if (map.containsKey(key)) {Node node = map.get(key);remove(node); // 从链表中移除当前节点addFirst(node); // 将当前节点移动到链表头部return node.value;}return -1;}// 写入缓存数据public void put(int key, int value) {if (map.containsKey(key)) {Node node = map.get(key);node.value = value;remove(node); // 从链表中移除当前节点addFirst(node); // 将当前节点移动到链表头部} else {if (map.size() == capacity) {map.remove(tail.prev.key); // 移除链表尾部节点对应的缓存数据remove(tail.prev); // 移除链表尾部节点}Node newNode = new Node(key, value);map.put(key, newNode);addFirst(newNode); // 将新节点添加到链表头部}}// 从链表中移除节点private void remove(Node node) {node.prev.next = node.next;node.next.prev = node.prev;}// 将节点添加到链表头部private void addFirst(Node node) {Node next = head.next;head.next = node;node.prev = head;node.next = next;next.prev = node;}
}

在这个示例中,LRUCache类实现了一个LRU缓存,使用双向链表来维护缓存数据的访问顺序,使用哈希表来快速查找缓存数据。通过get和put方法实现对缓存的读取和写入操作,保持缓存中数据的最近访问顺序。

这篇关于【退役之重学Java】Redis 的过期策略的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/978965

相关文章

Spring Boot 实现 IP 限流的原理、实践与利弊解析

《SpringBoot实现IP限流的原理、实践与利弊解析》在SpringBoot中实现IP限流是一种简单而有效的方式来保障系统的稳定性和可用性,本文给大家介绍SpringBoot实现IP限... 目录一、引言二、IP 限流原理2.1 令牌桶算法2.2 漏桶算法三、使用场景3.1 防止恶意攻击3.2 控制资源

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

springboot下载接口限速功能实现

《springboot下载接口限速功能实现》通过Redis统计并发数动态调整每个用户带宽,核心逻辑为每秒读取并发送限定数据量,防止单用户占用过多资源,确保整体下载均衡且高效,本文给大家介绍spring... 目录 一、整体目标 二、涉及的主要类/方法✅ 三、核心流程图解(简化) 四、关键代码详解1️⃣ 设置

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

SpringMVC高效获取JavaBean对象指南

《SpringMVC高效获取JavaBean对象指南》SpringMVC通过数据绑定自动将请求参数映射到JavaBean,支持表单、URL及JSON数据,需用@ModelAttribute、@Requ... 目录Spring MVC 获取 JavaBean 对象指南核心机制:数据绑定实现步骤1. 定义 Ja

javax.net.ssl.SSLHandshakeException:异常原因及解决方案

《javax.net.ssl.SSLHandshakeException:异常原因及解决方案》javax.net.ssl.SSLHandshakeException是一个SSL握手异常,通常在建立SS... 目录报错原因在程序中绕过服务器的安全验证注意点最后多说一句报错原因一般出现这种问题是因为目标服务器

Java实现删除文件中的指定内容

《Java实现删除文件中的指定内容》在日常开发中,经常需要对文本文件进行批量处理,其中,删除文件中指定内容是最常见的需求之一,下面我们就来看看如何使用java实现删除文件中的指定内容吧... 目录1. 项目背景详细介绍2. 项目需求详细介绍2.1 功能需求2.2 非功能需求3. 相关技术详细介绍3.1 Ja

springboot项目中整合高德地图的实践

《springboot项目中整合高德地图的实践》:本文主要介绍springboot项目中整合高德地图的实践,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一:高德开放平台的使用二:创建数据库(我是用的是mysql)三:Springboot所需的依赖(根据你的需求再

spring中的ImportSelector接口示例详解

《spring中的ImportSelector接口示例详解》Spring的ImportSelector接口用于动态选择配置类,实现条件化和模块化配置,关键方法selectImports根据注解信息返回... 目录一、核心作用二、关键方法三、扩展功能四、使用示例五、工作原理六、应用场景七、自定义实现Impor

SpringBoot3应用中集成和使用Spring Retry的实践记录

《SpringBoot3应用中集成和使用SpringRetry的实践记录》SpringRetry为SpringBoot3提供重试机制,支持注解和编程式两种方式,可配置重试策略与监听器,适用于临时性故... 目录1. 简介2. 环境准备3. 使用方式3.1 注解方式 基础使用自定义重试策略失败恢复机制注意事项