在 Golang 中實現 Cache::remember 方法詳解
項目需要把部分代碼移植到 Golang , 之前用 Laravel 封裝的寫起來很舒服,在 Golang 裡隻能自動動手實現.
一開始想的是使用 interface 實現,但是遇到瞭一個坑, Golang 裡的組合是一個虛假的繼承
package main import "fmt" type Person interface { Say() Name() } type Parent struct { } func (s *Parent) Say() { fmt.Println("i am " + s.Name()) } func (s *Parent) Name() string { return "parent" } type Child struct { Parent } func (s *Child) Name() string { return "child" } type Child1 struct { Parent } func main() { var c Child // i am parent c.Say() var c1 Child1 // i am parent c1.Say() }
- 如上 c.say() 代碼,在別的語言裡應該是輸出 i am child 才對, 而 Golang 不一樣,查瞭一下 Golang 的資料才能理解 https://golang.org/ref/spec#Selectors
- 大致意思是說,通過 x.f 調用 f 方法或者屬性時,從當前或者嵌套匿名結構體由淺到深的去調用,而不會去尋找上級
- 比如 child1 沒有 Say 方法,會進入到匿名結構體 Parent 找到 Say 方法,然後調用
- 而 child 也沒有 Say 方法,同樣去調用 Parent 的 Say 方法,這時候 Say 是通過 Parent 調用的, 當在 Say 裡調用 s.Name 方法,並不能找到 child , 所以還是會調用到 Parent 的 Name 方法
- 然後自己整理和同事一起寫瞭大致的 remember 方法
import ( "context" "encoding/json" "fmt" "github.com/gin-gonic/gin" "time" ) // redis 操作已經簡化 func CacheGet(c context.Context, t interface{}, cacheKey string, callQuery func() error) error { // 此處通過 redis 獲取數據, 如果存在數據, 那麼直接返回 dataBytes, err := redis.Get(c, cacheKey).Bytes() if err == nil { if err := json.Unmarshal(dataBytes, t); err == nil { return nil } } // 當 redis 沒有數據, 那麼調用此方法修改 t, if err := callQuery(); err != nil { return err } // 這裡把修改之後的 t 存儲到 redis, 下次使用便可以使用緩存 dataBytes, err = json.Marshal(t) if err == nil { redis.Set(c, cacheKey, dataBytes, time.Minute*30) } return nil } func handle(c *gin.Context) { var model models.User err := utils.CacheGet( c.Request.Context(), &model, fmt.Sprintf("cache_xxx:%s", c.Param("id")), func() error { return db.First(&model) }, ) }
到此這篇關於在 Golang 中實現 Cache::remember 方法的文章就介紹到這瞭,更多相關Golang實現 Cache::remember 內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- None Found