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

【Golang学习之旅】Go + Redis 缓存设计与优化(项目实战)

文章目录

    • 业务需求
      • 1. 缓存设计
      • 2. Go+Redis代码实现
      • 3. 代码解析
      • 4. 性能优化

在本节中,我们将通过一个实际的项目案例,演示如何在 Go 中使用 Redis 实现高效的缓存设计与优化。

业务需求

假设我们正在开发一个电商平台,需要缓存商品信息以提高页面加载速度。商品信息经常被查询,但修改频率较低,因此使用 Redis 作为缓存会大大减少数据库的查询压力。

1. 缓存设计

  1. 缓存粒度 :我们将每个商品的信息作为一个缓存项进行存储,缓存的键为商品的 ID,值为商品的 JSON 数据。
  2. 缓存失效策略:设置缓存的过期时间为 10 分钟,这样商品信息会在 10 分钟后自动失效,防止过期数据的出现。
  3. 缓存预热:在系统启动时,我们通过预加载常见商品的数据到缓存中,减少首次访问时的缓存未命中的情况。

2. Go+Redis代码实现

package main

import (
	"context"
	"encoding/json"
	"fmt"
	"log"
	"time"

	"github.com/go-redis/redis/v8"
)

var ctx = context.Background()

// 商品结构体
type Product struct {
	ID    string `json:"id"`
	Name  string `json:"name"`
	Price float64 `json:"price"`
}

func getProductFromDB(productID string) (*Product, error) {
	// 模拟从数据库获取商品数据
	return &Product{
		ID:    productID,
		Name:  "Example Product",
		Price: 99.99,
	}, nil
}

func getProductFromCache(rdb *redis.Client, productID string) (*Product, error) {
	// 从缓存中获取商品数据
	val, err := rdb.Get(ctx, productID).Result()
	if err == redis.Nil {
		// 缓存未命中,查询数据库
		return nil, nil
	} else if err != nil {
		return nil, err
	}

	var product Product
	err = json.Unmarshal([]byte(val), &product)
	if err != nil {
		return nil, err
	}
	return &product, nil
}

func setProductToCache(rdb *redis.Client, product *Product) error {
	// 将商品数据缓存到 Redis
	productData, err := json.Marshal(product)
	if err != nil {
		return err
	}

	return rdb.Set(ctx, product.ID, productData, 10*time.Minute).Err()
}

func getProduct(rdb *redis.Client, productID string) (*Product, error) {
	// 尝试从缓存中获取商品
	product, err := getProductFromCache(rdb, productID)
	if err != nil {
		return nil, err
	}

	if product == nil {
		// 缓存未命中,查询数据库并将结果缓存
		product, err = getProductFromDB(productID)
		if err != nil {
			return nil, err
		}

		err = setProductToCache(rdb, product)
		if err != nil {
			return nil, err
		}
	}

	return product, nil
}

func main() {
	rdb := redis.NewClient(&redis.Options{
		Addr:     "localhost:6379", // Redis 地址
		Password: "",               // 密码
		DB:       0,                // 默认数据库
	})

	// 获取商品
	productID := "12345"
	product, err := getProduct(rdb, productID)
	if err != nil {
		log.Fatalf("获取商品失败: %v", err)
	}

	fmt.Printf("商品信息: %+v\n", product)
}

3. 代码解析

  1. 获取商品信息:首先,我们尝试从 Redis 缓存中获取商品信息。如果缓存未命中,我们会从数据库中查询商品数据,并将查询结果存入缓存。
  2. 缓存设置过期时间:商品信息被存入缓存时,我们设置了 10 分钟的过期时间,这样缓存会自动失效。
  3. 缓存穿透与击穿防范:通过合理的缓存失效时间和商品信息的缓存设计,避免了缓存穿透和缓存击穿的问题。

4. 性能优化

通过上述设计,我们显著减少了数据库查询次数,降低了数据库的负载,提高了系统的响应速度和吞吐量。


http://www.kler.cn/a/539031.html

相关文章:

  • Android性能优化
  • 【LeetCode 刷题】贪心算法(4)-区间问题
  • python连点器
  • 正则表达式进阶(二)——零宽断言详解:\b \B \K \z \A
  • (七)C++的异常和文件输入与输出
  • RestTemplate Https 证书访问错误
  • 2.9学习总结
  • 从零开始了解人工智能:核心概念、GPT及 DeepSeek 探索
  • 使用cursor开发python调用deepseek本地模型实现本地AI对话
  • 如何学习多智能体系统协调(如自动驾驶车协同避让)
  • Linux:安装 node 及 nvm node 版本管理工具(ubuntu )
  • jvm view
  • 【LeetCode Hot100 堆】第 K 大的元素、前 K 个高频元素
  • 智慧城市节水管理信息系统项目解决方案
  • 在阿里云ECS上一键部署DeepSeek-R1
  • 7.Python文件操作:文件的打开与关闭、文件的读写、文件读写应用
  • 数据管理的“圣经”——《DAMA数据管理知识体系指南(第二版)》
  • 解锁 DeepSeek 模型高效部署密码:蓝耘平台深度剖析与实战应用
  • React 什么是抽象组件及为什么要抽象组件
  • 人工智能-A* 算法规划的路径进行动态调整
  • 分组加密算法CLEFIA
  • 【LLM】o1/R1系列LLM数据篇
  • 【开学补课复习专题】python 语言考试试题2
  • cuda学习资料汇总
  • 第六届MathorCup高校数学建模挑战赛-A题:淡水养殖池塘水华发生及池水自净化研究
  • C++ 实现封装的顺序表:顺序表的操作与实践