後端技術基本都要和緩存打交道,最近恰好工做上碰到了這方法的需求,迷迷糊糊的用了一下golang的這個包如今打算寫篇心得
緩存(cache),原始意義是指訪問速度比通常 隨機存取存儲器(RAM)快的一種高速存儲器,一般它不像系統主存那樣使用 DRAM技術,而使用昂貴但較快速的 SRAM技術。緩存的設置是全部現代計算機系統發揮高性能的重要因素之一。
打個不恰當的比喻,當你想要作菜,你須要有原材料,在對它進行處理,而後在吃。mysql
若是你每次切一次菜從冰箱裏拿一次,如此往復很是浪費時間,你本身也會累。這時候你能夠拿個籃子,一次將冰箱的你要用的材料(蔬菜)裝起來,而後放到菜板旁邊備用。git
上面的例子就是:冰箱:數據庫 蔬菜:數據 籃子:緩存容器 github
接下來就來介紹一下今天主要用的包 go-cachegolang
go-cache 是一個基於內存的、高速的,存儲k-v格式的緩存工具。它適用於運行在單臺機器上的應用程序,能夠存儲任何數據類型的值,並能夠被多個goroutine安全地使用。 雖然go-cache 不打算用做持久數據存儲,可是能夠將整個緩存數據保存到文件(或任何io.Reader/Writer)中,而且能快速從中指定數據源加載,快速恢復狀態。
type cache struct {
defaultExpiration time.Duration //默認的通用key實效時長
items map[string]Item //底層的map存儲
mu sync.RWMutex //因爲map是非線程安全的,增長的全局鎖
onEvicted func(string, interface{})//失效key時,回觸發,我本身命名爲回收函數
janitor *janitor //監視器,Goroutine,定時輪詢用於失效key
}複製代碼
demo算法
import (
"fmt"
"github.com/patrickmn/go-cache"
"time"
)
func main() {
// 默認過時時間爲5min,每10min清理一次過時緩存
c := cache.New(5*time.Minute, 10*time.Minute)
// 設置key-value,並設置爲默認過時時間
c.Set("foo", "bar", cache.DefaultExpiration)
// 設置一個不會過時的key,該key不會自動刪除,從新更新key或者使用c.Delete("baz")
c.Set("baz", 42, cache.NoExpiration)
// 從緩存獲取對應key的值
foo, found := c.Get("foo")
if found {
fmt.Println(foo)
}
// Since Go is statically typed, and cache values can be anything, type
// assertion is needed when values are being passed to functions that don't
// take arbitrary types, (i.e. interface{}). The simplest way to do this for
// values which will only be used once--e.g. for passing to another
// function--is:
foo, found := c.Get("foo")
if found {
MyFunction(foo.(string))
}
// This gets tedious if the value is used several times in the same function.
// You might do either of the following instead:
if x, found := c.Get("foo"); found {
foo := x.(string)
// ...
}
// or
var foo string
if x, found := c.Get("foo"); found {
foo = x.(string)
}
// ...
// foo can then be passed around freely as a string
// Want performance? Store pointers!
c.Set("foo", &MyStruct, cache.DefaultExpiration)
if x, found := c.Get("foo"); found {
foo := x.(*MyStruct)
// ...
}
}複製代碼
前面的例子對應到程序裏面就是我此次面對的一個小問題sql
我在網頁上須要導出一個報表,每行每一個單元格都須要從對應的model中取出相應的數據可是由於這項數據確定不可能只存在一個model裏,因此須要去查相關聯的表。若是每次用哪一個model就去庫裏去查相應的數據,速度就會巨慢無比(原來的人就是這麼寫的因此咱們須要去優化它)mongodb
首先咱們須要把數據從數據庫取出,這裏我用的是mongodb,也就是將裏面collection的數據所有取出來(collection你能夠理解爲mysql中的表)數據庫
for _, collection := range collections {
switch collection {
case "product":
products, err := gql.FindProduct(ctx, mongoplus.M{})
if err != nil {
logrus.Warn(err)
}
for _, product := range products {
temp := product
err := coll.Set("product_"+product.ID, &temp)
if err != nil {
logrus.Warn(err)
}
}
case "user":
users, err := gql.FindUser(ctx, mongoplus.M{})
if err != nil {
logrus.Warn(err)
}
for _, user := range users {
temp := user
err := coll.Set("user_"+user.ID, &temp)
if err != nil {
logrus.Warn(err)
}
}
case "company":
companys, err := gql.FindCompanyCache(ctx, mongoplus.M{})
if err != nil {
logrus.Warn(err)
}
for _, com := range companys {
temp := com
err := coll.Set("com_"+com.ID, &temp)
if err != nil {
logrus.Warn(err)
}
}
case "region":
Regions, err := gql.FindRegion(ctx, mongoplus.M{})
if err != nil {
logrus.Warn(err)
}
for _, Region := range Regions {
temp := Region
err := coll.Set("region_"+Region.ID, &temp)
if err != nil {
logrus.Warn(err)
}
}
case "industry":
industrys, err := gql.FindIndustry(ctx, mongoplus.M{})
if err != nil {
logrus.Warn(err)
}
for _, industry := range industrys {
temp := industry
err := coll.Set("industry_"+industry.ID, &temp)
if err != nil {
logrus.Warn(err)
}
}
}
}
return coll
}複製代碼
上面的代碼我把去出的數據都放在了容器裏面coll.Set("product_"+product.ID, &temp)
後端
我採用的方式是字段id_+model的形式緩存
而後要用的時候直接從數據庫中讀取數據就能優化一部分時間
gocache相對簡單,用了map[string]Item來進行存儲,沒有限制大小,只要內存容許能夠一直存,沒有上限,這個在實際生產中須要注意。
gocache很簡單,可是也有很多問題沒有作,簡單列一些本身想到的,能夠一塊兒優化下:
1. cache數量沒有上限,這個在線上使用的時候仍是容易出問題
2. 調用get獲取對象的時候,若是對象不存在,get方法會直接返回nil,其實這裏能夠優化一下若是沒有命中能夠從數據庫load一下。
三、一些命中沒法跟蹤。
其實對於go-cache還有其餘地方能夠分析,好比它鎖的粒度,結合系統的垃圾回收等等,可是因爲我學習golang語言時間不長,不少地方沒有學通就不寫出來丟人啦。等之後系統的學習go的多線程和其餘算法後,我會更新go-cache相關的知識。將來加油!