我是不会告诉你,我之前是手写缓存策略的

简介: gcache提供统一的缓存管理模块,提供了开发者可自定义灵活接入的缓存适配接口,并默认提供了高速内存缓存适配实现。

gcache提供统一的缓存管理模块,提供了开发者可自定义灵活接入的缓存适配接口,并默认提供了高速内存缓存适配实现。


先说结论


这篇文章通过结合商业项目的使用场景,为大家介绍了gcache的基本使用、缓存控制以及淘汰策略。

使用gcache做缓存处理,简单方便易上手!


优势


gcache模块默认提供的是一个高速的内存缓存,操作效率非常高效,CPU性能损耗在ns纳秒级别。使用简单易上手,非常适合单机应用使用。


基本使用


我们可以通过gcache.New()创建一个缓存对象

也可以直接使用gcache包方法,使用方式都是一样的。

下面代码段介绍了gcache的基本使用:


package main
import (
   "fmt"
   "github.com/gogf/gf/os/gcache"
   "github.com/gogf/gf/os/gtime"
)
func main() {
   // 创建一个缓存对象,
   c := gcache.New()
   // 设置缓存,不过期
   _ = c.Set("k1", "v1", 0)
   // 获取缓存
   v, _ := c.Get("k1")
   fmt.Println("k1对应的值:", v) //v1
   // 获取缓存大小
   n, _ := c.Size()
   fmt.Println("缓存大小:", n) //1
   // 缓存中是否存在指定键名
   b, _ := c.Contains("k1")
   fmt.Println("是否存在k1:", b)         //true
   isContains, _ := c.Contains("k2") //false
   fmt.Println("是否存在k2:", isContains)
   // 删除并返回被删除的键值
   fmt.Println(c.Remove("k1")) //v1
   // 关闭缓存对象,让GC回收资源
   _ = c.Close()
   // 当然也可以便捷地直接使用gcache包方法,使用方式和上面的一样
   _ = gcache.Set("k2", "v2", gtime.M*30) //半小时后到期
   k2Value, _ := gcache.Get("k2")
   fmt.Println("k2对应的值:", k2Value) //v2
}


打印结果


微信图片_20221113155145.jpg


缓存控制


下面介绍一些特殊场景的用法:

当键名不存在时写入:SetIfNotExist

根据键名取不到值时写入:GetOrSet


package main
import (
   "fmt"
   "github.com/gogf/gf/os/gcache"
   "time"
)
func main() {
   // 当键名不存在时写入,设置过期时间1秒
   _, _ = gcache.SetIfNotExist("k1", "v1", time.Second)
   // 打印当前的键名列表
   keys, _ := gcache.Keys()
   fmt.Println("打印当前的键名列表:", keys) //[k1]
   // 打印当前的键值列表
   values, _ := gcache.Values()
   fmt.Println("打印当前的键值列表:", values) //[v1]
   // 获取指定键值,如果不存在时写入,并返回键值;第三个参数设置为0表示永不过期
   v, _ := gcache.GetOrSet("k2", "v2", 0)
   fmt.Println("写入的键值:", v) //v2
   // 打印当前的键值对
   dataMap, _ := gcache.Data()
   fmt.Println("打印当前的键值对:", dataMap) // map[k1:v1 k2:v2]
   // 等待1秒,以便k1:v1自动过期
   time.Sleep(time.Second)
   // 再次打印当前的键值对,发现k1:v1已经过期,只剩下k2:v2
   data2, _ := gcache.Data()
   fmt.Println("等待一秒后,打印当前的键值对:", data2) // map[k2:v2]
}


打印结果


微信图片_20221113155148.jpg


缓存淘汰策略


之前做项目的时候也有这么设计过,现在gcache天然支持缓存淘汰策略


实战举例


给大家举个项目中的栗子:


比如我有一个新闻网站,为了保证新闻详情页的加载速度,会把新闻详情页做缓存处理(内容缓存,评论点赞等动态获取。)

一般来说新产生的新闻热度和访问量会更高,需要做缓存,而陈旧的新闻因为热度不再,访问量较低就从缓存中释放了。

解决上面的问题一般有2种思路:


  1. 设置缓存时间是一个解决思路
  2. 另外的思路就是设置缓存的淘汰策略。


比如:我一共允许缓存1万个缓存详情页,当超过1万个时,会淘汰掉最早缓存的数据,实现动态且稳定的缓存策略。

gcache天然支持这种缓存淘汰策略


代码示例


下面举个简单的栗子:


package main
import (
   "fmt"
   "github.com/gogf/gf/os/gcache"
   "time"
)
func main() {
   // 设置LRU淘汰数量
   c := gcache.New(2)
   // 添加10个元素,不过期
   for i := 0; i < 10; i++ {
      _ = c.Set(i, i, 0)
   }
   n, _ := c.Size()
   fmt.Println("缓存大小:", n)
   keys, _ := c.Keys()
   fmt.Println("缓存键值:", keys)
   // 读取键名1,保证该键名是优先保留
   v, _ := c.Get(1)
   fmt.Println("读取键名1的值:", v)
   // 等待一定时间后(默认1秒检查一次),
   // 元素会被按照从旧到新的顺序进行淘汰
   for i := 0; i < 10; i++ {
      time.Sleep(2 * time.Second)
      n, _ = c.Size()
      fmt.Println("缓存大小:", n)
      keys, _ = c.Keys()
      fmt.Println("缓存键值:", keys)
   }
}


打印结果


微信图片_20221113155151.jpg


小技巧


GetOrSetFunc的使用


GetOrSetFunc获取一个缓存值,当缓存不存在时执行指定的f func() (interface{}, error),缓存该f方法的结果值,并返回该结果。


总结


这篇文章通过结合商业项目的使用场景,为大家介绍了gcache的基本使用、缓存控制以及淘汰策略。

单机项目可以使用gcache做缓存处理,简单方便易上手;

分布式应用可以使用gredis做缓存处理,关注我,下一篇介绍gredis的使用。


公众号:程序员升级打怪之旅

微信号:wangzhongyang1993

B站视频:王中阳Go

相关文章
|
3月前
|
缓存 算法 数据挖掘
深入理解缓存更新策略:从LRU到LFU
【10月更文挑战第7天】 在本文中,我们将探讨计算机系统中缓存机制的核心——缓存更新策略。缓存是提高数据检索速度的关键技术之一,无论是在硬件还是软件层面都扮演着重要角色。我们会详细介绍最常用的两种缓存算法:最近最少使用(LRU)和最少使用频率(LFU),并讨论它们的优缺点及适用场景。通过对比分析,旨在帮助读者更好地理解如何选择和实现适合自己需求的缓存策略,从而优化系统性能。
81 3
|
1月前
|
缓存 API C#
C# 一分钟浅谈:GraphQL 中的缓存策略
本文介绍了在现代 Web 应用中,随着数据复杂度的增加,GraphQL 作为一种更灵活的数据查询语言的重要性,以及如何通过缓存策略优化其性能。文章详细探讨了客户端缓存、网络层缓存和服务器端缓存的实现方法,并提供了 C# 示例代码,帮助开发者理解和应用这些技术。同时,文中还讨论了缓存设计中的常见问题及解决方案,如缓存键设计、缓存失效策略等,旨在提升应用的响应速度和稳定性。
47 13
|
5月前
|
缓存 Java
Java本地高性能缓存实践问题之Caffeine缓存库中基于时间设置驱逐策略的问题如何解决
Java本地高性能缓存实践问题之Caffeine缓存库中基于时间设置驱逐策略的问题如何解决
|
27天前
|
存储 消息中间件 设计模式
缓存数据一致性策略如何分类?
数据库与缓存数据一致性问题的解决方案主要分为强一致性和最终一致性。强一致性通过分布式锁或分布式事务确保每次写入后数据立即一致,适合高要求场景,但性能开销大。最终一致性允许短暂延迟,常用方案包括Cache-Aside(先更新DB再删缓存)、Read/Write-Through(读写穿透)和Write-Behind(异步写入)。延时双删策略通过两次删除缓存确保数据最终一致,适用于复杂业务场景。选择方案需根据系统复杂度和一致性要求权衡。
49 0
|
2月前
|
存储 缓存 安全
在 Service Worker 中配置缓存策略
Service Worker 是一种可编程的网络代理,允许开发者控制网页如何加载资源。通过在 Service Worker 中配置缓存策略,可以优化应用性能,减少加载时间,提升用户体验。此策略涉及缓存的存储、更新和检索机制。
|
2月前
|
存储 缓存 监控
利用 Redis 缓存特性避免缓存穿透的策略与方法
【10月更文挑战第23天】通过以上对利用 Redis 缓存特性避免缓存穿透的详细阐述,我们对这一策略有了更深入的理解。在实际应用中,我们需要根据具体情况灵活运用这些方法,并结合其他技术手段,共同保障系统的稳定和高效运行。同时,要不断关注 Redis 缓存特性的发展和变化,及时调整策略,以应对不断出现的新挑战。
80 10
|
2月前
|
Web App开发 缓存 UED
如何设置浏览器的缓存策略?
【10月更文挑战第23天】通过合理地设置浏览器的缓存策略,可以在提高网页性能、减少网络流量的同时,确保用户能够获取到最新的内容,从而提升用户体验和网站的性能优化效果。
167 4
|
2月前
|
存储 消息中间件 缓存
缓存策略
【10月更文挑战第25天】在实际应用中,还需要不断地监控和调整缓存策略,以适应系统的变化和发展。
|
2月前
|
缓存 监控 NoSQL
Redis 缓存穿透及其应对策略
【10月更文挑战第23天】通过以上对 Redis 缓存穿透的详细阐述,我们对这一问题有了更深入的理解。在实际应用中,我们需要根据具体情况综合运用多种方法来解决缓存穿透问题,以保障系统的稳定运行和高效性能。同时,要不断关注技术的发展和变化,及时调整策略,以应对不断出现的新挑战。
63 4
|
3月前
|
存储 缓存 NoSQL
保持HTTP会话状态:缓存策略与实践
保持HTTP会话状态:缓存策略与实践