[Redis#5] hash | 命令 | 内部编码 | 应用 | cache: string, json, hash对比

news2024/11/26 6:16:19

目录

1 命令

HSET

HGET

HEXISTS

HDEL

HKEYS

HVALS

HGETALL

HMGET

HLEN

HSET NX

HINCRBY

HINCRBYFLOAT

2 命令小结

3 内部编码

4 使用场景

5 缓存方式对比

1. 原生字符串类型

2. 序列化字符串类型,例如 JSON 格式

3. 哈希类型


几乎所有的主流编程语言都提供了哈希类型,它们的叫法可能是哈希、字典、关联数组、映射。

在 Redis 中,哈希类型是指值本身又是一个键值对结构,形如 key = "key",value = {{field1, value1}, ..., {fieldN, valueN}}

Redis 键值对和哈希类型二者的关系可以通过图示表示,如下

可以将 字符串结构 优化 为 hash 结构,实现高内聚

对比

  • 字符串
    • 存储一个 uid 为 1 的用户对象,姓名 James,年龄 28
    • 示例:
user:1:name James
user:1:age 28
  • 哈希
    • 存储一个 uid 为 1 的用户对象,姓名 James,年龄 28
    • 示例:
user:1
name James
age 28

注意:哈希类型中的映射关系通常称为 field-value,用于区分 Redis 整体的键值对(key-value),这里的 value 是指 field 对应的值,而不是键(key)对应的值,请注意 value 在不同上下文的作用。

理解: Redis中的value的类型就是Hash类型的数据.也就是在Hash中又存储了一层Hash.


1 命令

HSET
  • 描述:设置 hash 中 指定的字段(field)的值(value)。
  • 语法HSET key field value [field value ...]
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:插入一组 field 为 O(1),插入 N 组 field 为 O(N)
  • 返回值:添加的字段的个数。
  • 示例
redis> HSET myhash field1 "Hello"
(integer) 1
redis> HGET myhash field1
"Hello"
HGET
  • 描述获取 hash 中指定字段的值。
  • 语法HGET key field
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(1)
  • 返回值:字段对应的值或者 nil。
  • 示例
redis> HSET myhash field1 "foo"
(integer) 1
redis> HGET myhash field1
"foo"
redis> HGET myhash field2
(nil)
HEXISTS
  • 描述:判断 hash 中是否 存在 指定的字段。
  • 语法HEXISTS key field
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(1)
  • 返回值1 表示存在,0 表示不存在。
  • 示例
redis> HSET myhash field1 "foo"
(integer) 1
redis> HEXISTS myhash field1
(integer) 1
redis> HEXISTS myhash field2
(integer) 0

注意:

  • 在hexists命令中,不支持一次性查询多个field,一次只可以查询 一个field.
  • 后面的 hmget 和 hmset 支持批量获取 和 设置
HDEL
  • 描述:删除 hash 中指定的字段。
  • 语法HDEL key field [field ...]
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:删除一个元素为 O(1),删除 N 个元素为 O(N)
  • 返回值:本次操作删除的字段个数。
  • 示例
redis> HSET myhash field1 "foo"
(integer) 1
redis> HDEL myhash field1
(integer) 1
redis> HDEL myhash field2
(integer) 0

HKEYS
  • 描述:获取 hash 中的 所有 field。
  • 语法HKEYS key
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(N),N 为 field 的个数。
  • 返回值:字段列表。
  • 示例
redis> HSET myhash field1 "Hello"
(integer) 1
redis> HSET myhash field2 "World"
(integer) 1
redis> HKEYS myhash
1) "field1"
2) "field2"
HVALS
  • 描述:获取 hash 中的 所有 value。
  • 语法HVALS key
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(N),N 为 field 的个数。
  • 返回值:所有的值。
  • 示例
redis> HSET myhash field1 "Hello"
(integer) 1
redis> HSET myhash field2 "World"
(integer) 1
redis> HVALS myhash
1) "Hello"
2) "World"
HGETALL
  • 描述:获取 hash 中的所有 field 以及对应的 value 。
  • 语法HGETALL key
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(N),N 为 field 的个数。
  • 返回值:字段和对应的值。
  • 示例
redis> HSET myhash field1 "Hello"
(integer) 1
redis> HSET myhash field2 "World"
(integer) 1
redis> HGETALL myhash
1) "field1"
2) "Hello"
3) "field2"
4) "World"

HMGET
  • 描述:一次获取 hash 中多个 field 的值。
  • 语法HMGET key field [field ...]
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:只查询一个元素为 O(1),查询多个元素为 O(N),N 为查询元素个数。
  • 返回值:字段对应的值或者 nil。
  • 示例
redis> HSET myhash field1 "Hello"
(integer) 1
redis> HSET myhash field2 "World"
(integer) 1
redis> HMGET myhash field1 field2 nofield
1) "Hello"
2) "World"
3) (nil)

注意

建议一次查多个减少网络请求

在使用 HGETALL 时,如果哈希元素个数比较多,会存在阻塞 Redis 的可能。

如果开发人员只需要获取部分 field,可以使用 HMGET,如果一定要获取全部 field,可以尝试使用 HSCAN 命令,该命令采用渐进式遍历哈希类型HSCAN 会在后续章节介绍。

HLEN
  • 描述:获取 hash 中的所有 field 的个数。
  • 语法HLEN key
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(1)
  • 返回值:字段个数。
  • 示例
redis> HSET myhash field1 "Hello"
(integer) 1
redis> HSET myhash field2 "World"
(integer) 1
redis> HLEN myhash
(integer) 2

HSET NX
  • 描述:在字段 不存在的情况下,设置 hash 中的字段和值。
  • 语法HSETNX key field value
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(1)
  • 返回值:1 表示设置成功,0 表示失败。
  • 示例
redis> HSETNX myhash field "Hello"
(integer) 1
redis> HSETNX myhash field "World"
(integer) 0
redis> HGET myhash field
"Hello"
HINCRBY
  • 描述:将 hash 中 field 对应的数值添加指定的值。
  • 语法HINCRBY key field increment
  • 命令有效版本:2.0.0 之后
  • 时间复杂度:O(1)
  • 返回值:该字段变化之后的值。
  • 示例
redis> HSET myhash field 5
(integer) 1
redis> HINCRBY myhash field 1
(integer) 6
redis> HINCRBY myhash field -1
(integer) 5
redis> HINCRBY myhash field -10
(integer) -5

HINCRBYFLOAT
  • 描述HINCRBY浮点数版本。
  • 语法HINCRBYFLOAT key field increment
  • 命令有效版本:2.6.0 之后
  • 时间复杂度:O(1)
  • 返回值:该字段变化之后的值。
  • 示例
redis> HSET mykey field 10.50
(integer) 1
redis> HINCRBYFLOAT mykey field 0.1
"10.6"
redis> HINCRBYFLOAT mykey field -5
"5.6"
redis> HSET mykey field 5.0e3
(integer) 0
redis> HINCRBYFLOAT mykey field 2.0e2
"5200"

2 命令小结

表 2-4 是哈希类型命令的效果、时间复杂度,开发人员可以参考此表,结合自身业务需求和数据大小选择合适的命令。

命令

执行效果

时间复杂度

hset key field value

设置值

O(1)

hget key field

获取值

O(1)

hdel key field [field ...]

删除 field

O(k),k 是 field 个数

hlen key

计算 field 个数

O(1)

hgetall key

获取所有的 field-value

O(k),k 是 field 个数

hmget field [field ...]

批量获取 field-value

O(k),k 是 field 个数

hmset field value [field value ...]

批量设置 field-value

O(k),k 是 field 个数

hexists key field

判断 field 是否存在

O(1)

hkeys key

获取所有的 field

O(k),k 是 field 个数

hvals key

获取所有的 value

O(k),k 是 field 个数

hsetnx key field value

设置值,但必须在 field 不存在时才能设置成功

O(1)

hincrby key field n

对应 field-value + n

O(1)

hincrbyfloat key field n

对应 field-value + n

O(1)

hstrlen key field

计算 value 的字符串长度

O(1)

3 内部编码

哈希的内部编码有两种:

ziplist(压缩列表)

  • 当哈希类型元素个数小于 hash-max-ziplist-entries 配置(默认 512 个),同时所有值都小于 hash-max-ziplist-value 配置(默认 64 字节)时,Redis 会使用 ziplist 作为哈希的内部实现。ziplist 使用更加紧凑的结构实现多个元素的连续存储
  • 所以在节省内存方面比 hashtable 更加优秀。

hashtable(哈希表)

  • 当哈希类型无法满足 ziplist 的条件时,Redis 会自动切换到使用哈希表作为哈希的内部实现。

优点:

  • 高效的读写:数据过多时 ziplist 的读写效率会下降,而 hashtable 的读写时间复杂度为 O(1)。,即使在数据量较大时也能保证高效的访问。
  • 良好的扩展性:适合存储大量数据和需要频繁更新的场景。

缺点:

  • 内存占用:相较于 ziplist,哈希表在内存使用上相对较多,特别是在存储小数据集时,内存开销更为显著。

哈希类型的内部编码,以及响应的变化

  • field个数⽐较少且没有⼤的value时,内部编码为ziplist
  • 当有value⼤于64字节时,内部编码会转换hashtable
  • field个数超过512时,内部编码也会转换hashtable

ps: 临界数值大小,无需记忆,是可以在 redis.conf 文件中配置的,主要是:理解这种优化思想~

对于 压缩算法的简单理解:


4 使用场景

关系型数据表保存用户信息

  • 结构:

映射关系表示用户信息

  • 用户信息表示为哈希类型:

  • 优势相⽐于使⽤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;
}

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

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


 

5 缓存方式对比

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

1. 原生字符串类型
  • 实现
set user:1:name James
set user:1:age 23
set user:1:city Beijing
  • 优点
    • 实现简单,针对个别属性变更也很灵活。
  • 缺点
    • 占用过多的键,内存占用量较大,同时用户信息在 Redis 中比较分散,缺少内聚性,所以这种方案基本没有实用性。
2. 序列化字符串类型,例如 JSON 格式
  • 实现
set user:1 经过序列化后的用户对象字符串
  • 优点
    • 针对总是整体作为操作的信息比较合适,编程也简单。同时,如果序列化方案选择合适,内存的使用效率很高。
  • 缺点
    • 本身序列化和反序列需要一定开销,同时如果总是操作个别属性则非常不灵活。
3. 哈希类型
  • 实现
hmset user:1 name James age 23 city Beijing
  • 优点
    • 简单、直观、灵活。尤其是针对信息的局部变更或者获取操作。
  • 缺点
    • 需要控制哈希在 ziplist 和 hashtable 两种内部编码的转换,可能会造成内存的较大消耗。
    • 不便于复杂的整体查询,没有规范的格式,较为稀疏,不好整体规划

💡end think : 人际交往也是:高内聚 低耦合...

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

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

相关文章

系统设计-高性能

一、CDN 1、什么是CDN CDN 就是将静态资源分发到多个不同的地方以实现就近访问&#xff0c;进而加快静态资源的访问速度&#xff0c;减轻服务器以及带宽的负担.(可看作是一层特殊缓存服务&#xff0c;用来处理静态资源的请求) 2、CDN工作原理 静态资源是如何被缓存到 CDN 节…

单片机知识总结(完整)

1、单片机概述 1.1. 单片机的定义与分类 定义&#xff1a; 单片机&#xff08;Microcontroller Unit&#xff0c;简称MCU&#xff09;是一种将微处理器、存储器&#xff08;包括程序存储器和数据存储器&#xff09;、输入/输出接口和其他必要的功能模块集成在单个芯片上的微型…

代码管理之Gitlab

文章目录 Git基础概述场景本地修改未提交&#xff0c;拉取远程代码修改提交本地&#xff0c;远程已有新提交 GitIDEA引入Git拉取仓库代码最后位置 Git基础 概述 workspace 工作区&#xff1a;本地电脑上看到的目录&#xff1b; repository 本地仓库&#xff1a;就是工作区中隐…

基于Java Springboot古风生活体验交流网站

一、作品包含 源码数据库设计文档万字PPT全套环境和工具资源部署教程 二、项目技术 前端技术&#xff1a;Html、Css、Js、Vue、Element-ui 数据库&#xff1a;MySQL 后端技术&#xff1a;Java、Spring Boot、MyBatis 三、运行环境 开发工具&#xff1a;IDEA/eclipse 数据…

智慧社区管理系统平台提升物业运营效率与用户体验

内容概要 智慧社区管理系统平台是一个集成了多项功能的综合性解决方案&#xff0c;旨在通过先进的技术手段提升物业管理的效率和居民的生活质量。该平台不仅关注物业运营的各个方面&#xff0c;还强调用户体验的重要性。随着科技的发展&#xff0c;社区管理方式正发生着翻天覆…

Docker--通过Docker容器创建一个Web服务器

Web服务器 Web服务器&#xff0c;一般指网站服务器&#xff0c;是驻留于因特网上某种类型计算机的程序。 Web服务器可以向浏览器等Web客户端提供文档&#xff0c;也可以放置网站文件以供全世界浏览&#xff0c;或放置数据文件以供全世界下载。 Web服务器的主要功能是提供网上…

健身房小程序服务渠道开展

健身不单单是锻炼身体、保持身材&#xff0c;也是一种社交方式&#xff0c;城市里门店不少&#xff0c;每家都有一定流量和老客&#xff0c;但仅靠传统线下拉客/自然流量前往和线上朋友圈、短视频发硬广等方式还不够。 商家需要找到更多潜在目标客户&#xff0c;而消费者也对门…

利用Java爬虫获得1688商品分类:技术解析与代码示例

在电商领域&#xff0c;了解商品分类对于市场分析和产品策略至关重要。1688作为中国领先的B2B电商平台&#xff0c;其商品分类数据对于商家来说具有极高的价值。本文将详细介绍如何使用Java编写爬虫程序&#xff0c;以合法合规的方式获取1688商品分类信息&#xff0c;并提供代码…

QT:QListView实现table自定义代理

介绍 QListVIew有两种切换形式&#xff0c;QListView::IconMode和QListView::ListMode&#xff0c;通过setViewMode()进行设置切换。因为QListView可以像QTreeView一样显示树形结构&#xff0c;也可以分成多列。这次目标是将ListView的ListMode形态显示为table。使用代理&…

YOLOv10改进,YOLOv10添加SE注意力机制,二次C2f结构

摘要 理论介绍 SE 注意力机制是一种提升卷积神经网络(CNN)性能的模块,SE更关注重要的特征图,增强了网络的表现,同时仅增加了较少的参数。SE 机制包含两个主要步骤: Squeeze (压缩):对所有特征图进行全局平均池化,生成一个通道描述符。Excitation (激励):将通道描述符…

详解Servlet的使用

目录 Servlet 定义 动态页面 vs 静态页面 主要功能 Servlet的使用 创建Maven项目 引入依赖 创建目录 编写代码 打war包 部署程序 验证程序 Smart Tomcat 安装Smart Tomcat 配置Smart Tomcat插件 启动Tomcat 访问页面 路径对应关系 Servlet运行原理 Tomcat的…

【Nginx从入门到精通】05-安装部署-虚拟机不能上网简单排错

文章目录 总结1、排查步骤 一、排查&#xff1a;Vmware网关二、排查&#xff1a;ipStage 1 &#xff1a;ping 127.0.0.1Stage 2 &#xff1a;ping 宿主机ipStage 3 &#xff1a;ping 网关 失败原因解决方案Stage 4 &#xff1a;ping qq.com 总结 1、排查步骤 Vmware中网关是否…

优化求解 | 非线性最小二乘优化器Ceres安装教程与应用案例

目录 0 专栏介绍1 Ceres库介绍2 Ceres库安装3 Ceres库概念3.1 构建最小二乘问题3.1.1 残差块3.1.2 代价函数 3.2 求解最小二乘问题 4 Ceres库案例4.1 Powell函数优化4.2 非线性曲线拟合 0 专栏介绍 &#x1f525;课设、毕设、创新竞赛必备&#xff01;&#x1f525;本专栏涉及…

Flink Transformation-转换算子

map算子的使用 假如有如下数据&#xff1a; 86.149.9.216 10001 17/05/2015:10:05:30 GET /presentations/logstash-monitorama-2013/images/github-contributions.png 83.149.9.216 10002 17/05/2015:10:06:53 GET /presentations/logstash-monitorama-2013/css/print/paper…

记录一些PostgreSQL操作

本文分享一些pg操作 查看版本 select version(); PostgreSQL 11.11 查看安装的插件 select * from pg_available_extensions; 查看分词效果 select ‘我爱北京天安门,天安门上太阳升’::tsvector; ‘天安门上太阳升’:2 ‘我爱北京天安门’:1select to_tsvector(‘我爱北京天…

【ubuntu】数学人的环境搭建

Python 语言环境 python 的 pip 第三方库管理 sudo apt install python3-pippython 的 idle 界面 sudo apt install idle3R 语言环境 sudo apt install r-cran-zoo### RStudio 界面 ubuntu sudo snap install rstudio --classicJulia 语言环境 sudo snap install julia --…

android 11添加切换分屏功能

引言 自Android 7开始官方就支持分屏显示,但没有切换分屏的功能,即交换上下屏幕。直到Android 13开始才支持切换分屏,操作方式是:分屏模式下双击中间分割线就会交换上下屏位置。本文的目的就是在Android 11上实现切换分屏的功能。 下图是Android13切换分屏演示 切换分屏…

数据结构——排序算法第一幕(插入排序:直接插入排序、希尔排序 选择排序:直接选择排序,堆排序)超详细!!!!

文章目录 前言一、排序1.1 概念1.2 常见的排序算法 二、插入排序2.1 直接插入排序2.2 希尔排序希尔排序的时间复杂度 三、选择排序3.1 直接选择排序3.2 堆排序 总结 前言 时间很快&#xff0c;转眼间已经到数据结构的排序算法部分啦 今天我们来学习排序算法当中的 插入排序 和 …

C++网络编程之多播

概述 在移动互联网时代&#xff0c;随着多媒体应用的日益普及&#xff0c;如何高效地将数据传输给多个接收者成为了网络通信领域的一个重要课题。多播&#xff08;英文为Multicast&#xff09;作为一种高效的网络通信方式&#xff0c;可以将数据同时发送到多个接收者&#xff0…

AWS的流日志

文章目录 一、aws如何观察vpc的日志&#xff1f;二、aws观测其vpc的入口日志三、 具体配置3.1、配置你的存储神器 S33.2、建立子网的流日志 一、aws如何观察vpc的日志&#xff1f; 排查问题的时候除了去抓包看具体的端口信息的时候&#xff0c;还可以根据其所在的vpc的子网信息…