Comment utiliser le contexte dans Go pour implémenter l'actualisation automatique du cache des résultats des requêtes
Résumé :
Dans le développement d'applications Web, afin d'améliorer l'expérience utilisateur, nous devons parfois mettre en cache les résultats de certaines requêtes pour réduire l'accès à la base de données ou autre services. Cependant, la période de validité des données mises en cache constitue un problème. Un cache expiré peut amener les utilisateurs à obtenir des données expirées, ce qui entraîne un affichage et des opérations erronés. Dans cet article, nous explorerons comment utiliser le package de contexte de Go pour implémenter la fonction d'actualisation automatique du cache des résultats de la demande afin de garantir l'actualité des données mises en cache.
type CacheItem struct { result interface{} expireAt time.Time } type Cache struct { cacheMap map[string]CacheItem mutex sync.RWMutex }
func (c *Cache) Get(key string) interface{} { c.mutex.RLock() defer c.mutex.RUnlock() item, ok := c.cacheMap[key] if ok && item.expireAt.After(time.Now()) { return item.result } // 发起请求并更新缓存 result := makeRequest(key) c.cacheMap[key] = CacheItem{result: result, expireAt: time.Now().Add(time.Minute)} return result }
func (c *Cache) RefreshCache(ctx context.Context, key string) { ticker := time.NewTicker(time.Minute) defer ticker.Stop() for { select { case <-ticker.C: result := makeRequest(key) c.mutex.Lock() c.cacheMap[key] = CacheItem{result: result, expireAt: time.Now().Add(time.Minute)} c.mutex.Unlock() case <-ctx.Done(): return } } }
package main import ( "context" "fmt" "net/http" "sync" "time" ) type CacheItem struct { result interface{} expireAt time.Time } type Cache struct { cacheMap map[string]CacheItem mutex sync.RWMutex } func makeRequest(key string) interface{} { // 模拟请求耗时 time.Sleep(time.Second) return fmt.Sprintf("result for %s", key) } func (c *Cache) Get(key string) interface{} { c.mutex.RLock() defer c.mutex.RUnlock() item, ok := c.cacheMap[key] if ok && item.expireAt.After(time.Now()) { return item.result } result := makeRequest(key) c.cacheMap[key] = CacheItem{result: result, expireAt: time.Now().Add(time.Minute)} return result } func (c *Cache) RefreshCache(ctx context.Context, key string) { ticker := time.NewTicker(time.Minute) defer ticker.Stop() for { select { case <-ticker.C: result := makeRequest(key) c.mutex.Lock() c.cacheMap[key] = CacheItem{result: result, expireAt: time.Now().Add(time.Minute)} c.mutex.Unlock() case <-ctx.Done(): return } } } func main() { cache := &Cache{cacheMap: make(map[string]CacheItem)} http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) { ctx, cancel := context.WithDeadline(context.Background(), time.Now().Add(time.Second*5)) defer cancel() key := r.URL.Path result := cache.Get(key) fmt.Fprintf(w, "%s: %s", key, result) // 启动刷新缓存的协程 go cache.RefreshCache(ctx, key) }) http.ListenAndServe(":8080", nil) }
Cet article explique comment utiliser le package de contexte de Go pour implémenter la fonction d'actualisation automatique du cache des résultats de la requête. En utilisant des structures de cache et des verrous mutex pour garantir la sécurité de la concurrence, et en utilisant les fonctionnalités du package de contexte pour actualiser régulièrement le cache, nous pouvons simplement mettre en cache les résultats de la requête et garantir l'actualité des données. L'exemple de code ci-dessus n'est qu'une simple démonstration. L'utilisation réelle peut nécessiter une modification et une optimisation appropriées en fonction de besoins spécifiques.
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!