Redis中Hash(哈希)类型的基本操作

news2024/11/14 15:30:54

文章目录

  • 一、 哈希简介
  • 二、常用命令
    • hset
    • hget
    • hexists
    • hdel
    • hkeys
    • hvals
    • hgetall
    • hmget
    • hlen
    • hsetnx
    • hincrby
    • hincrbyfloat
    • hstrlen
  • 三、命令小结
  • 四、哈希内部编码方式
  • 五、典型应用场景
  • 六、 字符串,序列化,哈希对比


一、 哈希简介

几乎所有的主流编程语言都提供了哈希(hash)类型,它们的叫法可能是哈希、字典、关联数组、映射。在 Redis 中,哈希类型是指值本身又是一个键值对结构,形如 key = “key”,value = { { field1, value1 }, …, {fieldN, valueN } },Redis 键值对和哈希类型二者的关系可以用图 2-15 来表示。
在这里插入图片描述
哈希类型中的映射关系通常称为 field-value,用于区分 Redis 整体的键值对(key-value)注意这里的 value 是指 field 对应的值,不是键(key)对应的值,请注意 value 在不同上下文的作用。

二、常用命令

hset

HSET

  • 设置 hash 中指定的字段(field)的值(value)。
  • 语法:HSET key field value [field value ...]
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:插入一组 field 为 O(1), 插入N组 field为 O(N)
  • 返回值:添加字段的个数。
  • 示例:
    在这里插入图片描述
    此处可以一次只插入一个,也可以一次插入N个。

hget

HGET

  • 获取 hash 中指定字段的值。
  • 语法: HGET key field
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:0(1)
  • 返回值:字段对应的值或者 nil。
  • 示例:

在这里插入图片描述

hexists

HEXISTS

  • 判断 hash 中是否有指定的字段。
  • 语法: HEXISTS key field
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(1)
  • 返回值:1表示存在,0表示不存在。
  • 示例:

在这里插入图片描述

hdel

HDEL

  • 删除 hash 中指定的字段。
  • 语法: HDEL key field [field ....]
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:删除一个元素为 O(1),删除 N 个元素为 O(N).
  • 返回值:本次操作删除的字段个数。
  • 示例:

在这里插入图片描述

hkeys

HKEYS

  • 获取 hash 中的所有字段(field)。
  • 语法: HKEYS key
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(N),N为field 的个数:
  • 返回值:字段列表。
  • 示例:

在这里插入图片描述

hvals

HVALS

  • 获取 hash 中的所有的值(value)。
  • 语法: HVALS key
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(N),N为 field 的个数.
  • 返回值:所有的值。
  • 示例:

在这里插入图片描述

hgetall

HGETALL

  • 获取 hash 中的所有字段(field)以及对应的值(value)。
  • 语法: HGETALL key
  • 命令有效版本:2.0.0之后
  • 时间复杂度:O(N),N为field 的个数,
  • 返回值:字段和对应的值。
  • 示例:
    在这里插入图片描述

hmget

HMGET

  • 一次获取 hash 中多个字段的值。
  • 语法: HMGET key field [field ...]
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:只查询一个元素为 0(1),查询多个元素为 O(N),N 为查询元素个数.
  • 返回值:字段对应的值或者 nil。
  • 示例:

在这里插入图片描述

在使用 HGETALL 时,如果哈希元素个数比较多,会存在阻塞 Redis 的可能。如果开发人员只需要获取部分 field,可以使用HMGET,如果一定要获取全部 field,可以尝试使用 HSCAN命令,该命令采用渐进式遍历哈希类型,HSCAN 会在后续介绍。

hlen

HLEN

  • 获取 hash 中的所有字段(field)的个数。
  • 语法: HLEN key
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(1)
  • 返回值:字段个数。
  • 示例:

在这里插入图片描述

hsetnx

HSETNX

  • 在字段不存在的情况下,设置 hash 中的字段和值,如果存在就不会设置。
  • 语法: HSETNX key field value
  • 命令有效版本:2.0.0之后
  • 时间复杂度:0(1)
  • 返回值:1表示设置成功,0 表示失败。
  • 示例:

在这里插入图片描述

hincrby

HINCRBY

  • 将 hash 中字段对应的数值添加指定的值。
  • 语法: HINCRBY key field increment
  • 命令有效版本:2.0.0之后
  • 时间复杂度:O(1)
  • 返回值:该字段变化之后的值。
  • 示例:

在这里插入图片描述

hincrbyfloat

HINCRBYFLOAT

  • HINCRBY的浮点数版本。
  • 语法: HINCRBYFLOAT key field increment
  • 命令有效版本:2.6.0之后
  • 时间复杂度:0(1)
  • 返回值:该字段变化之后的值。
  • 示例:

在这里插入图片描述

hstrlen

HSTRLEN

  • 计算 value 的字符串长度
  • 语法:HSTRLEN key field
  • 返回值:字符串的长度
  • 示例:

在这里插入图片描述

三、命令小结

命令执⾏效果时间复杂度
hset key field value设置值O(1)
hget key field获取值O(1)
hdel key field [field…]删除 fieldO(k), k 是 field 个数
hlen key计算 field 个数O(1)
hgetall key获取所有的 field-valueO(k), k 是 field 个数
hmget批量获取 field-valueO(k), k 是 field 个数
hmset批量设置 field-value(注意:Redis 3.0.6 后使用 hset 替代)O(k), k 是 field 个数
hexists key field判断 field 是否存在O(1)
hkeys key获取所有的 fieldO(k), k 是 field 个数
hvals key获取所有的 valueO(k), k 是 field 个数
hsetnx key field value设置值,但必须在 field 不存在时才能设置成功O(1)
hincrby key field n对应 field-value + n(n 为整数)O(1)
hincrbyfloat key field n对应 field-value + n(n 为浮点数)O(1)
hstrlen key field计算 value 的字符串长度O(1)

注意:hmset 命令在 Redis 3.0.6 版本后被废弃,推荐使用 hset 或 hmset 的变种(逐个设置)来替代批量设置的操作。表格中仍列出 hmset 以供参考。

四、哈希内部编码方式

哈希的内部编码有两种:

  • ziplist(压缩列表):当哈希类型元素个数小于 hash-max-ziplist-entries 配置(默认 512 个)同时所有值都小于 hash-max-ziplist-value配置(默认 64字节)时,Redis 会使用 ziplist 作为哈希的内部实现,ziplist 使用更加紧凑的结构实现多个元素的连续存储,所以在节省内存方面比hashtable 更加优秀。
  • hashtable(哈希表):当哈希类型无法满足 ziplist的条件时,Redis 会使用 hashtable 作为哈希的内部实现,因为此时 ziplist 的读写效率会下降,而 hashtable 的读写时间复杂度为 O(1)。下面的示例演示了哈希类型的内部编码,以及响应的变化。
  1. 当 field 个数比较少且没有大的 value 时,内部编码为 ziplist:

在这里插入图片描述

listpack 是 Redis 5.0 引入的一个新的内部数据结构,用于替代和优化 ziplist,但在 Redis的官方文档和上下文中,哈希类型仍然使用 ziplist 或 hashtable 作为其内部编码的术语。如果你在使用 Redis 5.0或更高版本,那么哈希类型在内部可能会使用 listpack 来实现 ziplist 的功能,但这一细节通常对开发者是透明的。

  1. 当有 value 大于 64 字节时,内部编码会转换为 hashtable:

在这里插入图片描述
3. 当 field 个数超过 512 时,内部编码也会转换为 hashtable:
由于field个数太多,博主也懒得敲了,感兴趣的自己试试吧。

五、典型应用场景

图 2-16 为关系型数据表记录的两条用户信息,用户的属性表现为表的列,每条用户信息表现为行。如果映射关系表示这两个用户信息,则如图 2-17 所示。
在这里插入图片描述
在这里插入图片描述
相比于使用 JSON 格式的字符串缓存用户信息,哈希类型变得更加直观,并且在更新操作上变得更灵活。可以将每个用户的 id 定义为键后缀,多对 field-value 对应用户的各个属性,类似如下伪代码:

UserInfo getUserInfo(long uid) {
	// 根据 uid 得到 Redis 的键
	String key = "user:" + uid;
	// 尝试从 Redis 中获取对应的值
	userInfoMap = Redis 执行命令:hgetall key;
	// 如果缓存命中(hit)
	if (value != null) {
		// 将映射关系还原为对象形式
		UserInfo userInfo = 利用映射关系构建对象(userInfoMap);
		return userInfo;
	}
	// 如果缓存未命中(miss)
	// 从数据库中,根据 uid 获取用户信息
	UserInfo userInfo = MySQL 执行 SQL:select * from user_info where uid = <uid>
		// 如果表中没有 uid 对应的用户信息
		if (userInfo == null) {
			响应 404
				return null;
		}
	// 将缓存以哈希类型保存
	Redis 执行命令:hmset key name userInfo.name age userInfo.age city userInfo.city
		// 写入缓存,为了防止数据腐烂(rot),设置过期时间为 1 小时(3600 秒)
		Redis 执行命令:expire key 3600
		// 返回用户信息
		return userInfo;
}

但是需要注意的是哈希类型和关系型数据库有两点不同之处:

  1. 哈希类型是稀疏的,而关系型数据库是完全结构化的,例如哈希类型每个键可以有不同的field,而关系型数据库一旦添加新的列,所有行都要为其设置值,即使为 null,如图 2-18所示。
  2. 关系数据库可以做复杂的关系查询,而 Redis 去模拟关系型复杂查询,例如联表查询、聚合查询等
    基本不可能,维护成本高。

图 2-18 关系型数据库稀疏性

六、 字符串,序列化,哈希对比

截至目前为止,我们已经能够用三种方法缓存用户信息,下面给出三种方案的实现方法和优缺点分析。

  1. 原生字符串类型 —— 使用字符串类型,每个属性一个键。
set user:1:name James
set user:1:age 23
set user:1:city Beijing

优点:实现简单,针对个别属性变更也很灵活。
缺点:占用过多的键,内存占用量较大,同时用户信息在 Redis 中比较分散,缺少内聚性,所以这种方案基本没有实用性。

  1. 序列化字符串类型,例如 JSON 格式
set user:1 经过序列化后的用户对象字符串

优点:针对总是以整体作为操作的信息比较合适,编程也简单。同时,如果序列化方案选择合适,内存的使用效率很高。
缺点:本身序列化和反序列需要一定开销,同时如果总是操作个别属性则非常不灵活。

  1. 哈希类型
hmset user:1 name James age 23 city Beijing

优点:简单、直观、灵活。尤其是针对信息的局部变更或者获取操作。
缺点:需要控制哈希在 ziplist 和 hashtable 两种内部编码的转换,可能会造成内存的较大消耗。

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

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

相关文章

(蓝桥杯)STM32G431RBT6(TIM4-PWM)

一、基础配置 这个auto-reload preload是自动重装载值&#xff0c;因为我们想让他每改变一个占空比&#xff0c;至少出现一次周期 Counter Period(Autoreload Regisiter)这个设值为10000&#xff0c;那么就相当于它的周期是10000 脉冲宽度可以设置为占周期的一半&#xff0c;那…

docker部署excalidraw画图工具

0&#xff09;效果 0.1&#xff09;实时协作 0.2&#xff09;导出格式 1&#xff09;docker安装 docker脚本 bash <(curl -sSL https://cdn.jsdelivr.net/gh/SuperManito/LinuxMirrorsmain/DockerInstallation.sh)docker-compose脚本 curl -L "https://github.com/…

【随手笔记】使用J-LINK读写芯片内存数据

第一种使用JLINK.exe 1. 打开j-link.exe 2.输入【usb】 3. 连接芯片 输入【connect】输入芯片型号【STM32L071RB】输入连接方式 【S】 使用SWD连接方式输入连接速率 【4000】连接成功 4. 输入【&#xff1f;】查看指令提示 5. 读写指令 Mem Mem [<Zone>…

Redis的主从模式、哨兵模式、集群模式

最近学习了一下这三种架构模式&#xff0c;这里记录一下&#xff0c;仅供参考 目录 一、主从架构 1、搭建方式 2、同步原理 3、优化策略&#xff1a; 4、总结&#xff1a; 二、哨兵架构 1、搭建哨兵集群 2、RedisTemplate如何使用哨兵模式 三、分片集群架构 1&#…

JVM面试题-说一下JVM主要组成部分及其作用

总体来说&#xff0c;方法区和堆是所有线程共享的内存区域&#xff1b;而虚拟机栈、本地方法栈和程序计数器的运行是线程私有的内存区域&#xff0c;运行时数据区域就是我们常说的JVM的内存。 类加载子系统&#xff1a;根据给定的全限定名类名(如&#xff1a;java.lang.Object…

【D3.js in Action 3 精译_024】3.4 让 D3 数据适应屏幕(上)

当前内容所在位置&#xff08;可进入专栏查看其他译好的章节内容&#xff09; 第一部分 D3.js 基础知识 第一章 D3.js 简介&#xff08;已完结&#xff09; 1.1 何为 D3.js&#xff1f;1.2 D3 生态系统——入门须知1.3 数据可视化最佳实践&#xff08;上&#xff09;1.3 数据可…

AJAX(一)HTTP协议(请求响应报文),AJAX发送请求,请求问题处理

文章目录 一、AJAX二、HTTP协议1. 请求报文2. 响应报文 三、AJAX案例准备1. 安装node2. Express搭建服务器3. 安装nodemon实现自动重启 四、AJAX发送请求1. GET请求2. POST请求(1) 配置请求体(2) 配置请求头 3. 响应JSON数据的两种方式(1) 手动&#xff0c;JSON.parse()(2) 设置…

AI绘画Flux【lora模型】【微缩景观】:惊艳!3D场景融入手机上的微景观!

大家好&#xff0c;我是灵魂画师向阳 今天和大家分享一款基于Flux底模训练的微缩景观模型——FLUX|手机上的微景观。此模型主要将手机作为微型景观的基底&#xff0c;强制将3d情景融入手机并控制在手机屏幕上方范围内。 作者在使用提示词直出和使用该Loar提示词生成的图片进行…

rsyslogd 内存占用很高解决方案

在Kubernetes&#xff08;K8S&#xff09;集群中&#xff0c;监控日志是非常重要的&#xff0c;而rsyslogd是Linux系统中用于处理系统和应用程序日志的守护进程。有时候rsyslogd可能会占用较高的内存&#xff0c;这时候我们就需要对其进行优化和调整。 阿里云虚拟服务器&…

JavaEE: 深入探索TCP网络编程的奇妙世界(二)

文章目录 TCP核心机制TCP核心机制二: 超时重传为啥会丢包?TCP如何对抗丢包?超时重传的时间设定超时时间该如何确定? TCP核心机制 书接上文~ TCP核心机制二: 超时重传 在网络传输中,并不会一帆风顺,而是可能出现"丢包情况"~ 为啥会丢包? 产生丢包的原因有很多…

其他比较条件

使用BETWEEN条件 可以用BETWEEN范围条件显示基于一个值范围的行。指定的范围包含一个下限和一个上限。 示例&#xff1a;查询employees表&#xff0c;薪水在3000-8000之间的雇员ID、名字与薪水。 select employee_id,last_name,salary from employees where salary between 3…

移植Linux:如何制作rootfs?

一、分析 1. 文件系统简介 理论上说一个嵌入式设备如果内核能够运行起来&#xff0c;且不需要运行用户进程的话&#xff0c;是不需要文件系统的&#xff0c;文件系统简单的说就是一种目录结构&#xff0c;由于 linux操作系统的设备在系统中是以文件的形式存在&#xff0c;将这…

Cypress安装与启动(开始学习记录)

一 Cypress安装 使用npm安装 1.查看node.js npm的版本&#xff0c;输入 npm --version 和 node --version&#xff0c;node.js没安装的可以去中文网下载最新稳定版安装&#xff0c;npm不建议升级到最新版本&#xff0c;会导致安装Cypress时Error: Cannot find module ansi-st…

2-94 基于matlab的最佳维纳滤波器的盲解卷积算法

基于matlab的最佳维纳滤波器的盲解卷积算法。维纳滤波将地震子波转换为任意所需要的形态。维纳滤波不同于反滤波&#xff0c;它是在最小平方的意义上为最 佳。基于最佳纳滤波理论的滤波器算法是莱文逊(Wiener—Levinson)算法。程序提供了4种子波和4种期望输出&#xff1a;零延迟…

JavaEE: 深入探索TCP网络编程的奇妙世界(一)

文章目录 TCPTCP协议段落格式TCP相关机制TCP核心机制一: 确认应答32位序号32位确认序号后发先至问题 TCP TCP要比UDP更复杂一些~ TCP的全称为"传输控制协议".他负责对数据的传输进行一个详细的控制. TCP协议段落格式 源/目的端口号: 表示数据是从哪个进程来.到哪个…

Innodb内存结构

缓冲池Buffer Pool: 缓冲池是innodb内存结构缓冲区中的核心部分&#xff0c;在服务启动的时候服务器会向操作系统申请一块大小为128MB的内存空间&#xff0c;所有对数据库中数据的增删查改操作均在缓冲池bufferPool中完成&#xff0c;并且缓冲区中其他组件的描述信息也都存储在…

[Linux]Vi和Vim编辑器

Vi和Vim编辑器 Linux系统会内置vi文本编辑器, 类似于windows中的记事本 Vim具有程序编辑的能力, 可以看作是Vi的增强版本, 可以进行语法检查, 代码补全,代码编译和错误调整等功能 Vi和Vim的模式 快速入门 使用vim开发一个Hello.java程序 通过Xshell连接Linux系统命令行输入…

技术美术百人计划 | 《4.5 DOF景深算法》笔记

1. 景深定义 景深&#xff08;Depth of Field&#xff0c;DOF&#xff09;&#xff0c;是指在摄影机镜头或其他成像器前沿能够取得清晰图像的成像所测定的被摄物体前后距离范围。镜头光圈、镜头焦距、及焦平面到拍摄物的距离是影响景深的重要因素。在聚焦完成后&#xff0c;焦点…

2024年Q3国际信息系统安全认证联盟(ISC2)内部研讨会要点分享

2024年是CISSP认证成立30周年&#xff0c;这是一项具有里程碑意义的成就&#xff0c;代表了CISSP在网络安全领域的卓越、创新和领导力。博主于今年9月份参加了ISC2&#xff08;国际信息系统安全认证联盟&#xff09;组织的2024年第3季度内部网络研讨会&#xff0c;针对会议中的…

【sgCreateCallAPIFunction】自定义小工具:敏捷开发→调用接口方法代码生成工具

<template><div :class"$options.name" class"sgDevTool"><sgHead /><div class"sg-container"><div class"sg-start"><div style"margin-bottom: 10px">调用接口方法定义列表</div…