当前位置: 首页 > news >正文

【Redis_Day6】Hash类型

【Redis_Day6】Hash类型

  • Hash类型
  • 操作hash的命令
    • hset:设置hash中指定的字段(field)的值(value)
    • hsetnx:想hash中添加字段并设置值
    • hget:获取hash中指定字段的值
    • hexists:判断hash中是否有指定的字段
    • hdel:删除hash中指定的字段
    • hkeys:获取hash中所有的字段
    • hvals:获取hash中所有的值
    • hgetall:获取hash中所有字段和其对应的值
    • hmget:一次获取hash中多个字段的值
    • hlen:获取hash中所有字段的个数
    • hincrby:给hash中指定字段对应的值 加n(n是整数)
    • hincrbyfloat:给hash中指定字段对应的值 加n(n是浮点数)
    • hash相关命令小结
  • hash内部编码
  • hash的应用
    • 作为缓存
      • 三种缓存方式对比
  • 高内聚低耦合

Hash类型

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

在Redis中,哈希类型是指值本⾝⼜是⼀个键值对结构,形如key="key",value={{field1,value1},…,{fieldN,valueN}}。
哈希类型中的映射关系通常称为field-value,⽤于区分Redis整体的键值对(key-value),使用过程中要注意value在不同上下⽂的作⽤。

如下是字符串和哈希类型的对比:
在这里插入图片描述
redis自身就是键值对结构。
redis自身的键值对就是通过hash的方式来组织的。
针对一个键值对来说,其中的value同样可以是键值对,该键值对的类型是hash。

操作hash的命令

使用下列h系列命令的前提:key对应的value必须是hash类型。

hset:设置hash中指定的字段(field)的值(value)

总结hset的用法:
hset key field value [field value ...]:设置key的值,值的类型是hash。对一个key,可以一次给它设置多个field-value。返回值是添加的字段field的个数。如果key不存在,先创建key。
field-value中的value要求是字符串类型。
在这里插入图片描述在hset命令中,设置字段field1的值时,如果field1存在,那么本次设置的value会覆盖field1原来的值。

hsetnx:想hash中添加字段并设置值

hsetnx命令也是给字段设置值,但是要求字段不存在才能设置成功,也就是在hash中新建字段field1并给它设置值。

总结hsetnx的用法:
hsetnx key field1 value:key中不存在field1,添加field1并设置它的值为value,并返回1。如果field已存在,则操作无效,返回0。
命令成功前提是key存在且key中不存在field1。

在这里插入图片描述

hget:获取hash中指定字段的值

总结hget的用法:
hget key1 field1:获取key1中field1对应的value。返回值是fileld1对应的值或nil。
在这里插入图片描述

hexists:判断hash中是否有指定的字段

总结hexists的用法:
exists key field:去key对应的hash类型的value中找,有没有字段field。找到了返回1,没找到返回0。
在这里插入图片描述

hdel:删除hash中指定的字段

总结hdel的用法:
hdel key field [field ...]:删除key中指定的field,可以一次性删除多个。返回值是本次操作成功删除的字段个数。
在这里插入图片描述

hdel命令是删除key中指定的字段,通用命令del是删除key。

hkeys:获取hash中所有的字段

总结hkeys的用法:
hkeys key:根据key找到对应的hash,然后遍历该hash,获取hash中的所有字段,返回值是字段列表。
在这里插入图片描述

hvals:获取hash中所有的值

总结hvals的用法:
hvals key:根据key找到对应的hash,然后遍历hash,获取hash中所有的值。返回值是值列表。
在这里插入图片描述
hkeys和hvals命令操作起来都是存在一定风险的。这两个命令的时间复杂度都是O(N),N是哈希的元素个数,如果哈希非常大,执行这两个命令的时间会很长,会导致redis服务器堵塞,无法给其他客户端提供服务。

hgetall:获取hash中所有字段和其对应的值

总结hgetall的用法:
hgetall key:遍历key对应的value中所有的field-value,并输出。
在这里插入图片描述hgetall命令也是危险的操作。
在使⽤hgetall时,如果哈希元素个数⽐较多,会存在阻塞Redis的可能。如果开发⼈员只需要获取部分field,可以使⽤hmget,如果⼀定要获取全部field,可以尝试使⽤hscan命令
hscan是渐进式遍历,即hscan命令遍历hash的时候,每次只遍历hash中的一部分,连续执行多次hscan命令,才能完整整个遍历。hkeys,hvals,hgetall三个命令遍历hash的时候都是一次性遍历完全部hash。

hmget:一次获取hash中多个字段的值

总结hmget的用法:
hmget key field [field ...]: 一次获取hash中多个指定字段的值。返回值是对应字段的值或nil。
在这里插入图片描述

hlen:获取hash中所有字段的个数

总结hlen的用法:
hlen key:获取hash中所有字段的个数,返回值是字段的个数。
在这里插入图片描述执行hlen命令是不用遍历的。

hincrby:给hash中指定字段对应的值 加n(n是整数)

field-value中的value也可以当做数字来处理。

总结hincrby的用法:
hincrby key field incrment:给field对应的value 加上incrment。返回值是field的新值。
在这里插入图片描述

hincrbyfloat:给hash中指定字段对应的值 加n(n是浮点数)

hincrbyfloat是hincrby的浮点数版本。

hash相关命令小结

在这里插入图片描述
在这里插入图片描述
其中,hstrlen key field计算的是field-value中的value的长度。

hash内部编码

hash的内部编码有两种,分别是ziplist(压缩列表)hashtable(哈希表)

压缩的本质是针对数据进行重新编码。
不同的数据,有不同的特点,结合这些特点进行精妙的设计(压缩算法),重新编码之后,就能缩小体积。比如一个字符串abc0000000000000efg,重新编码后字符串变成了abc13[13]efg。

zipllist内部的数据结构也是精心设计的,相比于hashtable存储hash会存在浪费空间的问题,redis内部用ziplist类型存储hash,能节省内存空间。

ziplist的缺点,读写元素时候速度较慢,如果hash中元素个数少,慢的不明显,但是如果元素个数太多,就很慢了。

所以如果哈希中的元素个数比较少,使用ziplist存储,如果hash中元素个数比较多,使用hashtable来存储。
且如果field-value中value都比较短,使用ziplist存储hash,如果某个value的长度过长,也会转换成hashtable存储hash。

通过修改redis配置文件redis.conf中的配置项hash-max-ziplist-entries(默认512个),确定hash中的元素个数超过多少时用hashtable存储hash。
通过修改redis配置文件redis.conf中的配置项hash-max-ziplist-value(默认64字节),确定hash中的元素长度超过多少时用hashtable存储hash。

在这里插入图片描述

hash的应用

作为缓存

redis+mysql组成的缓存存储架构:
在这里插入图片描述
假设把最近使用到的数据作为热点数据存储在redis中,
在上面的场景中:
redis做缓存,应用服务器访问数据,先查询Redis,如果redis上数据存在,就直接从redis上取数据交给应用服务器,不继续访问mysql了。如果redis上数据不存在,再读取mysql,把读到的结果返回给应用服务器的同时,把这个数据也写入到redis中。

用伪代码模拟上述场景,理解hash在其中的应用:
redis用key-value组织用户信息,
key是user:uid,value是hash类型。
在这里插入图片描述
用mysql保存用户信息:
在这里插入图片描述

//当用户发来请求时,假设业务根据用户uid获取用户信息:
//业务层
UserInfo getUserInfo(long uid){//先查询redis,假设用户信息保存在"user:uid"对应的键中。String key = "user:"+uid;//从redis中获取对应的值UserInfoMap value = redis命令:hgetall key;//如果缓存命中(hit)if(value != null){//将映射关系还原成对象形式UserInfo userInfo =  利用映射关系构建对象(UserInfoMap value);return userInfo;}//如果缓存没有命中(miss)//从数据库中,根据uid获取用户信息UserInfo userInfo = mysql执行sql语句:select * from user_info where uid = <uid>;//如果表中没有 uid 对应的⽤⼾信息if(userInfo == null){响应404return null;}//将缓存以哈希类型保存redis执行命令:hmset user:userInfo.uid name userInfo.name age userInfo.age city userInfo.city//写入缓存,为了防止数据腐烂(rot),设置过期时间为1 ⼩时(3600 秒)redis执行命令:expire user:userInfo.uid 3600//返回用户信息return userInfo;
}

比较hash类型和关系型数据库存储用户信息的区别:

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

三种缓存方式对比

  1. 原生字符串类型⸺⸺使用字符串类型,每个属性⼀个键。
    在这里插入图片描述优点:实现简单,针对个别属性变更也很灵活。
    缺点:占用过多的键,内存占用量较⼤,同时用户信息在Redis中比较分散,缺少内聚性,所以这种方案基本没有实⽤性。

  2. 序列化字符串类型,例如JSON格式
    在这里插入图片描述优点:针对总是以整体作为操作的信息比较合适,编程也简单。同时,如果序列化方案选择合适,内存的使用效率很高。
    缺点:本⾝序列化和反序列需要⼀定开销,同时如果总是操作个别属性则非常不灵活。

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

比较redis用String类型(Json格式)和用hash类型存储用户信息的区别:

如果用String(JSON)的格式来表示UserInfo,从redis里取出的数据要先通过JSON反序列化转换成对象,从mysql取出的数据要先通过JSON序列化转换成JSON格式。

但是如果用hash类型保存UserInfo,就不用上面两个格外的步骤。且用field表示对象的每个属性,还可以非常方便的修改/获取任何一个属性的值。
当然用hash类型保存UserInfo也有缺点,使用hash类型需要控制hash在ziplist和hashtable两种内部编码的转换,这可能会造成内存的较大消耗。(用空间换时间)

高内聚低耦合

内聚和耦合是衡量代码好坏的标准之一。好的代码应该是高内聚低耦合的。

高内聚:内聚是用来衡量一个模块内部各个元素(如函数、类的方法等)之间的功能相关性和紧密程度的。高内聚意味着模块内的各个部分在功能上高度相关,共同协作完成一个明确的任务。相反,低内聚则表示模块内的各个部分功能不相关或关系松散。
比如上面提到的,把一个用户的所有信息放到hash中一个key中,就是高内聚代码。

低耦合:耦合指的是两个代码/模块之间的关联关系,好的代码追求的是低耦合。模块之间耦合度越高,模块之间越容易相互影响。
比如在分布式系统中引入消息队列就是一个低耦合应用的例子:
假设在一个分布式系统中,A服务器直接调用B服务器,A给B发送请求,B处理请求返回响应给A,此时A和B之间的耦合度是比较大的。
但如果在A和B之间引入消息队列,此时A,消息队列,B三者构成生产者消费者模型,A把请求发给消息队列,B再从消息队列中获取请求,这种情况下如果A出现问题,牵连到B的可能性就很小,A和B之间互连程度降低,即降低了A和B之间的耦合度。


下次见~
在这里插入图片描述

相关文章:

【Redis_Day6】Hash类型

【Redis_Day6】Hash类型 Hash类型操作hash的命令hset&#xff1a;设置hash中指定的字段&#xff08;field&#xff09;的值&#xff08;value&#xff09;hsetnx&#xff1a;想hash中添加字段并设置值hget&#xff1a;获取hash中指定字段的值hexists&#xff1a;判断hash中是否…...

[开源] SafeLine 好用的Web 应用防火墙(WAF)

SafeLine&#xff0c;中文名 “雷池”&#xff0c;是一款简单好用, 效果突出的 Web 应用防火墙(WAF)&#xff0c;可以保护 Web 服务不受黑客攻击 一、简介 雷池通过过滤和监控 Web 应用与互联网之间的 HTTP 流量来保护 Web 服务。可以保护 Web 服务免受 SQL 注入、XSS、 代码注…...

40分钟学 Go 语言高并发:Select多路复用

Select多路复用 学习目标 知识点掌握程度应用场景select实现原理深入理解底层机制channel通信和多路选择超时处理掌握超时控制方法避免阻塞和资源浪费优先级控制理解优先级实现处理多个channel的顺序性能考虑了解性能优化点高并发场景优化 1. Select实现原理 让我们通过一个…...

candence: 如何快速设置SUBCLASS 的颜色

如何快速设置SUBCLASS 的颜色 一、一般操作 正常情况下修改SUBCLASS&#xff0c;需要如下步骤进行设置&#xff1a; 二、快速操作 右键&#xff0c;选择一个颜色即可...

FinalShell进行前端项目部署及nginx配置

首先需要准备服务器(阿里云、腾讯云都可)与域名&#xff1b; 示例为阿里云服务器&#xff1b; 1.进行FinalShell下载 下载官网 https://www.hostbuf.com/ 2.下载完毕后 配置FinalShell ssh ​ 名称自定义即可&#xff01; 2-1 提示连接成功 ​ 3.首先检查nginx是否下载 …...

神经网络(系统性学习一):入门篇——简介、发展历程、应用领域、基本概念、超参数调优、网络类型分类

相关文章&#xff1a; 神经网络中常用的激活函数 神经网络简介 神经网络&#xff08;Neural Networks&#xff09;是受生物神经系统启发而设计的数学模型&#xff0c;用于模拟人类大脑处理信息的方式。它由大量的节点&#xff08;或称为“神经元”&#xff09;组成&#xff0…...

用nextjs开发时遇到的问题

这几天已经基本把node后端的接口全部写完了&#xff0c;在前端开发时考虑时博客视频类型&#xff0c;考虑了ssr&#xff0c;于是选用了nextJs&#xff0c;用的是nextUi,tailwincss,目前碰到两个比较难受的事情。 1.nextUI个别组件无法在服务器段渲染 目前简单的解决方法&…...

微前端基础知识入门篇(二)

概述 在上一篇介绍了一些微前端的基础知识,详见微前端基础知识入门篇(一)。本文主要介绍qiankun微前端框架的实战入门内容。 qiankun微前端实践 通过Vite脚手架分别创建三个程序,主应用A为:vite+vue3+ts,两个微应用分别为B:vite+vue3+ts;C:vite+React+ts。因为qiankun的…...

自然语言处理:第六十五章 MinerU 开源PDF文档解析方案

本人项目地址大全&#xff1a;Victor94-king/NLP__ManVictor: CSDN of ManVictor 原文地址&#xff1a;MinerU&#xff1a;精准解析PDF文档的开源解决方案 论文链接&#xff1a;MinerU: An Open-Source Solution for Precise Document Content Extraction git地址&#xff1…...

Arcpy 多线程批量重采样脚本

Arcpy 多线程批量重采样脚本 import arcpy import os import multiprocessingdef resample_tifs(input_folder, output_folder, cell_size0.05, resampling_type"BILINEAR"):"""将指定文件夹下的所有 TIFF 文件重采样到指定分辨率&#xff0c;并输出…...

python 画图例子

目录 多组折线图点坐标的折线图 多组折线图 数据: 第1行为x轴标签第2/3/…行等为数据,其中第一列为标签&#xff0c;后面为y值 图片: 代码: import matplotlib.pyplot as plt# 原始数据字符串 # 第1行为x轴标签 # 第2/3/...行等为数据,其中第一列为标签&#xff0c;后面…...

Win11 22H2/23H2系统11月可选更新KB5046732发布!

系统之家11月22日报道&#xff0c;微软针对Win11 22H2/23H2版本推送了2024年11月最新可选更新补丁KB5046732&#xff0c;更新后&#xff0c;系统版本号升至22621.4541和22631.4541。本次更新后系统托盘能够显示缩短的日期和时间&#xff0c;文件资源管理器窗口很小时搜索框被切…...

【STM32】MPU6050初始化常用寄存器说明及示例代码

一、MPU6050常用配置寄存器 1、电源管理寄存器1&#xff08; PWR_MGMT_1 &#xff09; 此寄存器允许用户配置电源模式和时钟源。 DEVICE_RESET &#xff1a;用于控制复位的比特位。设置为1时复位 MPU6050&#xff0c;内部寄存器恢复为默认值&#xff0c;复位结束…...

深度学习中的mAP

在深度学习中&#xff0c;mAP是指平均精度均值(mean Average Precision)&#xff0c;它是深度学习中评价模型好坏的一种指标(metric)&#xff0c;特别是在目标检测中。 精确率和召回率的概念&#xff1a; (1).精确率(Precision)&#xff1a;预测阳性结果中实际正确的比例(TP / …...

Redis设计与实现 学习笔记 第二十章 Lua脚本

Redis从2.6版本引入对Lua脚本的支持&#xff0c;通过在服务器中嵌入Lua环境&#xff0c;Redis客户端可以使用Lua脚本&#xff0c;直接在服务器端原子地执行多个Redis命令。 其中EVAL命令可以直接对输入的脚本进行求值&#xff1a; 而使用EVALSHA命令则可以根据脚本的SHA1校验…...

大模型(LLMs)推理篇

大模型&#xff08;LLMs&#xff09;推理篇 1. 为什么大模型推理时显存涨的那么多还一直占着&#xff1f; 首先&#xff0c;序列太长了&#xff0c;有很多Q/K/V&#xff1b;其次&#xff0c;因为是逐个预测next token&#xff0c;每次要缓存K/V加速解码。 大模型在gpu和cpu上…...

Leetcode 412. Fizz Buzz

Problem Given an integer n, return a string array answer (1-indexed) where: answer[i] “FizzBuzz” if i is divisible by 3 and 5.answer[i] “Fizz” if i is divisible by 3.answer[i] “Buzz” if i is divisible by 5.answer[i] i (as a string) if none of t…...

双因子认证:统一运维平台安全管理策略

01双因子认证概述 双因子认证&#xff08;Two-Factor Authentication&#xff0c;简称2FA&#xff09;是一种身份验证机制&#xff0c;它要求用户提供两种不同类型的证据来证明自己的身份。这通常包括用户所知道的&#xff08;如密码&#xff09;、用户所拥有的&#xff08;如…...

CMake笔记:install(TARGETS target,...)无法安装的Debug/lib下

1. 问题描述 按如下CMake代码&#xff0c;无法将lib文件安装到Debug/lib或Release/lib目录下&#xff0c;始终安装在CMAKE_INSTALL_PREFIX/lib下。 install(TARGETS targetCONFIGURATIONS DebugLIBRARY DESTINATION Debug/lib) install(TARGETS targetCONFIGURATIONS Release…...

使用ENSP实现NAT

一、项目拓扑 二、项目实现 1.路由器AR1配置 进入系统试图 sys将路由器命名为R1 sysname R1关闭信息中心 undo info-center enable进入g0/0/0接口 int g0/0/0将g0/0/0接口IP地址配置为12.12.12.1/30 ip address 12.12.12.1 30进入e0/0/1接口 int g0/0/1将g0/0/1接口IP地址配置…...

web vue 项目 Docker化部署

Web 项目 Docker 化部署详细教程 目录 Web 项目 Docker 化部署概述Dockerfile 详解 构建阶段生产阶段 构建和运行 Docker 镜像 1. Web 项目 Docker 化部署概述 Docker 化部署的主要步骤分为以下几个阶段&#xff1a; 构建阶段&#xff08;Build Stage&#xff09;&#xff1a…...

调用支付宝接口响应40004 SYSTEM_ERROR问题排查

在对接支付宝API的时候&#xff0c;遇到了一些问题&#xff0c;记录一下排查过程。 Body:{"datadigital_fincloud_generalsaas_face_certify_initialize_response":{"msg":"Business Failed","code":"40004","sub_msg…...

Lombok 的 @Data 注解失效,未生成 getter/setter 方法引发的HTTP 406 错误

HTTP 状态码 406 (Not Acceptable) 和 500 (Internal Server Error) 是两类完全不同的错误&#xff0c;它们的含义、原因和解决方法都有显著区别。以下是详细对比&#xff1a; 1. HTTP 406 (Not Acceptable) 含义&#xff1a; 客户端请求的内容类型与服务器支持的内容类型不匹…...

相机Camera日志实例分析之二:相机Camx【专业模式开启直方图拍照】单帧流程日志详解

【关注我&#xff0c;后续持续新增专题博文&#xff0c;谢谢&#xff01;&#xff01;&#xff01;】 上一篇我们讲了&#xff1a; 这一篇我们开始讲&#xff1a; 目录 一、场景操作步骤 二、日志基础关键字分级如下 三、场景日志如下&#xff1a; 一、场景操作步骤 操作步…...

ElasticSearch搜索引擎之倒排索引及其底层算法

文章目录 一、搜索引擎1、什么是搜索引擎?2、搜索引擎的分类3、常用的搜索引擎4、搜索引擎的特点二、倒排索引1、简介2、为什么倒排索引不用B+树1.创建时间长,文件大。2.其次,树深,IO次数可怕。3.索引可能会失效。4.精准度差。三. 倒排索引四、算法1、Term Index的算法2、 …...

leetcodeSQL解题:3564. 季节性销售分析

leetcodeSQL解题&#xff1a;3564. 季节性销售分析 题目&#xff1a; 表&#xff1a;sales ---------------------- | Column Name | Type | ---------------------- | sale_id | int | | product_id | int | | sale_date | date | | quantity | int | | price | decimal | -…...

JUC笔记(上)-复习 涉及死锁 volatile synchronized CAS 原子操作

一、上下文切换 即使单核CPU也可以进行多线程执行代码&#xff0c;CPU会给每个线程分配CPU时间片来实现这个机制。时间片非常短&#xff0c;所以CPU会不断地切换线程执行&#xff0c;从而让我们感觉多个线程是同时执行的。时间片一般是十几毫秒(ms)。通过时间片分配算法执行。…...

2023赣州旅游投资集团

单选题 1.“不登高山&#xff0c;不知天之高也&#xff1b;不临深溪&#xff0c;不知地之厚也。”这句话说明_____。 A、人的意识具有创造性 B、人的认识是独立于实践之外的 C、实践在认识过程中具有决定作用 D、人的一切知识都是从直接经验中获得的 参考答案: C 本题解…...

学校时钟系统,标准考场时钟系统,AI亮相2025高考,赛思时钟系统为教育公平筑起“精准防线”

2025年#高考 将在近日拉开帷幕&#xff0c;#AI 监考一度冲上热搜。当AI深度融入高考&#xff0c;#时间同步 不再是辅助功能&#xff0c;而是决定AI监考系统成败的“生命线”。 AI亮相2025高考&#xff0c;40种异常行为0.5秒精准识别 2025年高考即将拉开帷幕&#xff0c;江西、…...

如何在网页里填写 PDF 表格?

有时候&#xff0c;你可能希望用户能在你的网站上填写 PDF 表单。然而&#xff0c;这件事并不简单&#xff0c;因为 PDF 并不是一种原生的网页格式。虽然浏览器可以显示 PDF 文件&#xff0c;但原生并不支持编辑或填写它们。更糟的是&#xff0c;如果你想收集表单数据&#xff…...