redis基本数据结构-hash

news2025/1/5 17:38:45

这里写自定义目录标题

    • 1. redis的数据结构hash
      • 1.1 Hash 数据结构的特点
      • 1.2 常见命令
      • 1.3 适用示例
    • 2. 常见业务场景
      • 2.1 用户信息存储
        • 2.1.1 场景
        • 2.1.2 优势
        • 2.1.3 解决方案
        • 2.1.4 代码实现
      • 2.2 购物车管理
        • 2.2.1 背景
        • 2.2.2 优势
        • 2.2.3 解决方案
        • 2.2.4 代码实现
    • 3. 注意事项:

1. redis的数据结构hash

参考链接:https://mp.weixin.qq.com/s/srkd73bS2n3mjIADLVg72A
Redis 的 Hash 数据结构是一种键值对集合,它非常适合表示对象或实体的属性。每个 Hash 可以存储多个字段,每个字段则对应一个值,这使得 Hash 成为一种高效的存储和查询方式,尤其是在处理对象数据时。

1.1 Hash 数据结构的特点

  • 节省空间: 对于存储多个字段的小对象,Hash 可以节省内存,因为 Redis 会对小的 Hash 结构进行内部优化。
  • 方便操作: 可以单独操作 Hash 中的字段,而不需要序列化整个对象。这个相较于string数据结构而已的显著特点。
  • 快速访问: 通过字段名快速访问对应的值,适合频繁读取和更新的场景。

1.2 常见命令

以下是一些常用的 Redis Hash 命令:
HSET: 设置 Hash 中指定字段的值。
HGET: 获取 Hash 中指定字段的值。
HMSET: 设置多个字段的值。
HMGET: 获取多个字段的值。
HGETALL: 获取 Hash 中所有字段和值。
HDEL: 删除 Hash 中指定字段。
HINCRBY: 对 Hash 中的字段值进行增减。
HEXISTS: 检查 Hash 中是否存在指定字段。

1.3 适用示例

# 创建用户 1000 的 Hash
HSET user:1000 name "Alice" age 30 city "New York"
# 查询用户姓名
HGET user:1000 name
# 返回 "Alice"
# 更新用户城市
HSET user:1000 city "Los Angeles"
# 查询所有用户信息
HGETALL user:1000
# 返回 {"name": "Alice", "age": "30", "city": "Los Angeles"}
# 增加用户年龄
HINCRBY user:1000 age 1
# 查询用户年龄
HGET user:1000 age
# 返回 "31"
# 删除用户年龄字段
HDEL user:1000 age

2. 常见业务场景

  1. 用户信息存储: 在社交网络或电商平台中,每个用户的基本信息(如姓名、地址、年龄等)可以使用 Hash 存储,方便快速读取和更新。
  2. 商品属性管理: 在电商平台中,商品的属性(如名称、价格、库存等)可以存储为 Hash,便于管理和查询。
  3. 会话管理: 在 Web 应用中,可以使用 Hash 来存储用户的会话信息,如登录状态、购物车内容等。
  4. 统计信息: 对于需要频繁更新的统计数据(如访问量、点赞数等),可以使用 Hash 来存储并进行原子性增减操作。
  5. 配置管理: 使用 Hash 存储应用程序的配置项,方便快速修改和读取。

2.1 用户信息存储

2.1.1 场景

在社交网络应用中,每个用户都有一系列属性,如用户名、年龄、兴趣爱好等。使用Hash类型可以方便地存储和查询单个用户的详细信息。

2.1.2 优势
  1. 结构化存储:将用户信息以字段和值的形式存储,易于理解和操作。
  2. 快速读写:Redis的Hash操作提供高速的读写性能。
  3. 灵活更新:可以单独更新用户信息中的某个字段,而无需重新设置整个对象。
2.1.3 解决方案

使用Redis Hash类型来存储和管理用户信息。当用户信息更新时,只更新Hash中的对应字段。
在这里插入图片描述

2.1.4 代码实现
package main

import (
    "context"
    "fmt"
    "github.com/go-redis/redis/v8"
    "log"
)

var ctx = context.Background()

// Redis 客户端初始化
var rdb = redis.NewClient(&redis.Options{
    Addr:     "",  // Redis 服务器地址
    Password: "", // 密码
    DB:       0,                  // 使用默认 DB
})

// 存储用户信息到 Redis Hash
func storeUserInfo(userID string, userInfo map[string]interface{}) {
    hashKey := "user:" + userID
    // 将用户信息存储到 Redis 的 Hash 中
    err := rdb.HSet(ctx, hashKey, userInfo).Err()
    if err != nil {
       log.Fatalf("Error storing user info: %v", err)
    }
}

// 从 Redis Hash 获取用户信息
func getUserInfo(userID string) map[string]string {
    hashKey := "user:" + userID
    // 从 Redis 获取用户信息
    fields, err := rdb.HGetAll(ctx, hashKey).Result()
    if err != nil {
       log.Fatalf("Error getting user info: %v", err)
       return nil
    }
    return fields
}

func main() {
    // 示例用户信息
    userID := "12345"
    userInfo := map[string]interface{}{
       "username":  "john_doe",
       "age":       30,
       "interests": "coding, hiking, reading",
    }

    // 存储用户信息
    storeUserInfo(userID, userInfo)

    // 获取并显示用户信息
    retrievedUserInfo := getUserInfo(userID)
    fmt.Printf("Retrieved User Info: %+v\n", retrievedUserInfo)
}

在这里插入图片描述

2.2 购物车管理

2.2.1 背景

在电商平台中,用户的购物车需要记录用户选择的商品及其数量。使用Hash类型可以有效地管理每个用户的购物车。

2.2.2 优势
  1. 快速添加和修改:可以快速添加商品到购物车或更新商品数量。
  2. 批量操作:可以一次性获取或更新购物车中的多个商品。
2.2.3 解决方案

使用Redis Hash类型来实现购物车功能,每个用户的购物车作为一个独立的Hash存储。
在这里插入图片描述

2.2.4 代码实现
package main

import (
    "context"
    "fmt"
    "github.com/go-redis/redis/v8"
    "log"
    "strconv"
)

var ctx = context.Background()

// Redis 客户端初始化
var rdb = redis.NewClient(&redis.Options{
    Addr:     "",  // Redis 服务器地址
    Password: "", // 密码
    DB:       0,                  // 使用默认 DB
})

// 添加商品到购物车
func addToCart(cartID string, productID string, quantity int) {
    cartKey := "cart:" + cartID
    // 使用 HINCRBY 命令增加商品数量
    err := rdb.HIncrBy(ctx, cartKey, productID, int64(quantity)).Err()
    if err != nil {
       log.Fatalf("Error adding to cart: %v", err)
    }
}

// 获取购物车中的商品和数量
func getCart(cartID string) map[string]int {
    cartKey := "cart:" + cartID
    // 从 Redis 获取购物车内容
    items, err := rdb.HGetAll(ctx, cartKey).Result()
    if err != nil {
       log.Fatalf("Error getting cart: %v", err)
       return nil
    }
    // 将商品 ID 和数量转换为映射
    cart := make(map[string]int)
    for productID, quantity := range items {
       qty, err := strconv.Atoi(quantity)
       if err != nil {
          log.Printf("Error converting quantity for product %s: %v", productID, err)
          continue
       }
       cart[productID] = qty
    }
    return cart
}

func main() {
    // 示例购物车 ID 和商品
    cartID := "user:123:cart"
    addToCart(cartID, "product:1", 2)
    addToCart(cartID, "product:2", 3)
    addToCart(cartID, "product:1", 1) // 增加产品 1 的数量

    // 获取并显示购物车内容
    cart := getCart(cartID)
    fmt.Println("Shopping Cart Contents:")
    for productID, quantity := range cart {
       fmt.Printf("Product ID: %s, Quantity: %d\n", productID, quantity)
    }
}

在这里插入图片描述
也可以根据hash的命令扩展功能。

3. 注意事项:

  • Hash类型的字段值可以是字符串,最大容量为512MB。
  • 在并发环境下,应确保对Hash的操作是线程安全的,可以使用事务或Lua脚本来保证。
  • 存储较大的Hash时,应注意性能和内存使用情况,合理设计数据结构以避免过度膨胀。
  • 定期清理和维护Hash数据,避免数据冗余和失效数据的累积。
    简单用lua脚本的形式:
// 添加商品到购物车(使用 Lua 脚本)
func addToCartWithLua(cartID string, productID string, quantity int) {
    cartKey := "cart:" + cartID

    // Lua 脚本:增加购物车中商品的数量   
     luaScript := `
        local cartKey = KEYS[1]
        local productID = ARGV[1]
        local quantity = tonumber(ARGV[2])
        return redis.call('HINCRBY', cartKey, productID, quantity)
    `

    // 执行 Lua 脚本   
     _, err := rdb.Eval(ctx, luaScript, []string{cartKey}, productID, quantity).Result()
    if err != nil {
        log.Fatalf("Error executing Lua script: %v", err)
    }
}

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

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

相关文章

使用虚拟信用卡WildCard轻松订阅POE:全面解析平台功能与订阅方式

POE(Platform of Engagement)是一个由Quora推出的人工智能聊天平台,汇集了多个强大的AI聊天机器人,如GPT-4、Claude、Sage等。POE提供了一个简洁、统一的界面,让用户能够便捷地与不同的AI聊天模型进行互动。这种平台的…

Shadertoy和desmos用来快速图像化辅助计算的好工具

Desmos适用场景解直线方程例子 Shadertoy是一个专门通过shader片段利用gpu像素着色的工具。每一帧都会执行显示区域每个像素点的着色。默认片段坐标是左下角(0,0)到右上角(像素分辨率大小)。有网页版,也有vscode插件版。插件版更方便.如果要验证一些图像化的计算。…

MyBatis-Plus分页查询、分组查询

目录 准备工作1. 实体类2. Mapper类3. 分页插件4. 数据 分页查询1. 使用条件构造器2. 使用自定义sql 分组查询1. 分组结果类2. 自定义sql3. 测试类 准备工作 1. 实体类 对地址字段address使用字段类型转换器,将List转为字符串数组保存在数据库中 package com.exa…

(web自动化测试+python)1

一.UI自动化测试介绍 1.测试化理论 UI就是指的是用户接口,指的是用户与电脑的接口,是用户界面 UI不仅仅指的是web,还可以指代app 我们为什么要进行自动化? 大量版本的回归 当新的功能出现,复测之间的--我们叫做回归&am…

《Diffusion Models Without Attention》CVPR2024

摘要 这篇论文探讨了在高保真图像生成领域,去噪扩散概率模型(Denoising Diffusion Probabilistic Models, DDPMs)的重要性。尽管DDPMs在捕捉复杂视觉分布方面表现出色,但在高分辨率图像生成上面临显著的计算挑战。现有的方法&…

动物目标检测——基于YOLOv5和树莓派4B平台

目标检测在计算机视觉领域中具有重要意义。YOLOv5(You Only Look One-level)是目标检测算法中的一种代表性方法,以其高效性和准确性备受关注,并且在各种目标检测任务中都表现出卓越的性能。本文将详细介绍如何在性能更强的计算机上…

java实习生第一次被分配需求——完成需求的大概流程

一、分配需求后第一步,首先是把项目跑起来 在我进入公司一两个星期之后(基本熟悉了公司的框架),就被我所在的开发小组的某个大哥分派了一个需求,然后他给我发了一个git地址,以及一个git的分支(…

知名专家曹启富主任:冠心病低龄化?早预防早受益,守护心脏从日常做起

冠心病,这一曾被视为中老年疾病的代表,如今正悄然向更年轻的人群逼近。冠心病多发生于40岁以上的年龄段,但近年来,其发病低龄化的趋势日益明显,根据数据显示,我国城市人群15岁以及15岁以上的人口&#xff0…

STM32G474之TAMPALRM输出

TAMPALRM输出源是指“RTC唤醒”、“RTC报警A”和“RTC报警B”输出,可以配置从RTC_OU1(PC13)或RTC_OUT2(PB2)输出,而OUT2EN用来决定从哪个引脚输出。 1、TAMPALRM输出原理见下表: 若不看CALIB输出,可以简化如下表: 2、 …

【API Testing and Development with Postman 2nd_001】关于本书

译者按 今天又淘到一本介绍 Postman 的宝藏级小册子,非常适合想进一步了解 API 接口测试的朋友们。本书最大的特点就是手把手教学。想当年第 1 版问世时,初出茅庐的我随便拣了书中一两招,就能轻松搞定工作中五花八门的 API 疑难杂症。只是当时…

监听键盘事件

问题:点击输入框弹出键盘遮挡文字 需求:点击输入框键盘弹起,点击别处键盘回收,输入框回到原来状态, 解决办法: 1.采用占位的思想(隐藏),文本框控制采用焦点控制&#…

多态的概念

多态 所谓的多态其实就是多种形态,它又被分为编译时多态(静态多态) 和 运行时多态(动态多态)。 静态的多态其实就是之前的模版和函数重载,今天我们主要讲动态的多态。所谓的动态多态其实就是相同的函数,完成不同的功能。 这就实现了明明都是…

C#搭建WebApi服务

1,OWIN的介绍 OWIN 的全称是 "Open Web Interface for .NET", OWIN 在 .NET Web 服务器和 .NET Web 应用之间定义了一套标准的接口, 其目的是为了实现服务器与应用之间的解耦,使得便携式 .NET Web 应用以及跨平台的愿望…

MongoDB事务机制

事务机制 1.事务概念 在对数据的操作的过程中,涉及到一连串的操作,这些操作如果失败,会导致我们的数据部分变化了,部分没变化。这个过程就好比如你去吃早餐,你点完餐了,并且吃完早餐了,没付钱你…

ES6标准---【五】【看这一篇就够了!!!】

目录 ES6以往文章 箭头函数的基本用法 箭头函数的用处 简化回调函数 rest参数与箭头函数结合 箭头函数使用注意点 this指向的问题 其它不存在的变量 不能使用call()、apply()、bind()方法改变this的指向 箭头函数不适用场合 定义对象时,对象方法内部包含…

信创环境下源代码防泄露解决方案

在当今数字化时代,信息安全已成为企业生存与发展的基石,尤其是在信息技术应用创新(信创)环境下,数据保护更是被提升至前所未有的高度。SDC沙盒防泄露系统以其独特的技术架构和卓越的安全性能,在信创环境中构…

ES6标准---【六】【学习ES6标准看这一篇就够了!!!】

目录 以往ES6文章 前言 对象属性的简洁表示法 一个实际例子 简介写法在打印对象时也很有用 注意 对象属性名表达式 用表达式做属性名 用表达式定义方法名 注意 对象方法的name属性 对象属性的可枚举性和遍历 可枚举性 属性的遍历 属性比那里次序规则 super关键…

图片生成PPT!首推这款一站式AI制作PPT工具!

在当今快节奏的工作中,制作一份精美的PPT演示文稿往往是一项费时费力的工作,特别是当我们需要将大量的图片转化为PPT时,传统的方法显得尤为繁琐。幸运的是,随着AI人工智能技术的飞速发展,一种更便捷地将图片转为ppt的解…

计算机毕业设计 《计算机基础》网上考试系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍:✌从事软件开发10年之余,专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ 🍅文末获取源码联系🍅 👇🏻 精…

Java之线程篇三

​​​​​​​ 目录 线程状态 观察线程的所有状态 线程状态及其描述 线程状态转换 代码示例1 代码示例2 线程安全 概念 线程不安全的代码示例 线程不安全的原因 线程安全的代码示例-加锁 synchronized关键字 synchronized的特性 小结 形成死锁的四个必要条件 …