Maison développement back-end Golang Implémenter un cache LRU dans Go

Implémenter un cache LRU dans Go

Aug 05, 2024 pm 04:04 PM

Implement an LRU Cache in Go

Vous avez donc besoin d'un petit cache et ne pouvez pas justifier une instance Redis ou Memcached. Voyons ce qu'il faut pour en implémenter un dans Go. Pour le plaisir, nous le réaliserons en utilisant des génériques afin qu'il soit réutilisable dans notre projet.

Un cache LRU a généralement une capacité fixe et la politique d'éjection la plus simple : éjecter l'élément qui a le plus de temps depuis son accès. Un simple cache lru implémentera l'interface suivante :

type LRUCache[T any] interface {
    Get(key string) (value T, found bool)
    Put(key string, value T)
    Keys() []string
    Remove(key string) bool
    Clear()
    Capacity() int
    Len() int
}
Copier après la connexion

Nous savons que le cache stockera un élément de données sous la forme d'une entrée saisie par une certaine valeur. Cela ressemble à une carte. Qu’en est-il de la mise en œuvre de la politique d’éjection ? Une façon de procéder consiste à conserver une propriété timeAccessed avec chaque élément. Quelque chose comme :

type cacheEntry[T any] struct {
  Data T
  LastAccessed time.time
}
Copier après la connexion

Cependant, pensons aux performances, nous voulons pouvoir rechercher la clé de cache ainsi qu'insérer et expulser la plus ancienne, si nécessaire, le plus rapidement possible.

L'utilisation d'une carte, qui est une table de hachage, nous donnera des performances assez rapides pour les recherches. Et si vous trouviez l'entrée la plus ancienne ? Si votre structure de cache ressemble à :

type LRUCache[T any] {
  capacity int
  keyMap map[string]cacheEntry[T]
}
Copier après la connexion

Il faudra forcément parcourir la carte pour retrouver la plus ancienne lorsque viendra le temps d'expulser une entrée.

Nous avons besoin d'un moyen de stocker les entrées d'une manière qui nous permette efficacement de conserver une liste des entrées du cache triées. Il est préférable que nous n'ayons pas besoin d'utiliser une routine de tri.

Une liste à double lien est un bon moyen de le faire et nous n'avons pas besoin de stocker le temps d'accès dans l'entrée à moins que nous le souhaitions réellement. Supposons donc que nous ayons une liste chaînée qui implémente ce qui suit avec sa structure de nœud :

type DoubleLinkedList[T any] interface {
    Head() *DoubleNode[T]
    Tail() *DoubleNode[T]
    // Append inserts new item at tail
    Append(data T) *DoubleNode[T]
    // Push appends new item at head
    Push(data T) *DoubleNode[T]
    Remove(node *DoubleNode[T]) *DoubleNode[T]
    RemoveTail() *DoubleNode[T]
    MoveToHead(node *DoubleNode[T])
}
type DoubleNode[T any] struct {
    Data T
    Prev *DoubleNode[T]
    Next *DoubleNode[T]
}
Copier après la connexion

La structure du cache peut maintenant ressembler à :

type lruCache[T any] struct {
    capacity int
    keyMap   map[string]*DoubleNode[lruCacheEntry[T]]
    list     DoubleLinkedList[lruCacheEntry[T]]
}
Copier après la connexion

La structure d'entrée du cache sera :

type lruCacheEntry[T any] struct {
    key   string
    value T
}
Copier après la connexion

En réalité, vous utiliseriez probablement une interface pour la clé de cache. J'utilise une chaîne pour garder le code simple.

Dans l'implémentation ici, l'entrée du cache la plus récemment consultée sera en tête et la moins récemment utilisée sera en queue. Ainsi, quand vient le temps d’expulser, nous supprimons simplement l’élément tail de la liste chaînée.

L'implémentation de la fonction Get() est simple :

func (l *lruCache[T]) Get(key string) (value T, found bool) {
    if node, ok := l.keyMap[key]; ok {
        l.list.MoveToHead(node)
        return node.Data.value, ok
    }
    var zero T
    return zero, false
}
Copier après la connexion

Get a juste besoin de récupérer l'entrée de carte pour la clé, puis de déplacer le nœud en tête de la liste puisqu'il est désormais le « le plus récemment utilisé ».

La fonction Put() est l'endroit où nous gérerons l'expulsion si cela est nécessaire :

func (l *lruCache[T]) Put(key string, value T) {
    if node, ok := l.keyMap[key]; ok {
        node.Data = lruCacheEntry[T]{
            key:   key,
            value: value,
        }
        // move the element to the most recent position
        l.list.MoveToHead(node)
    } else {
        // insert the new element at the head
        newNode := l.list.Push(lruCacheEntry[T]{
            key:   key,
            value: value,
        })
        l.keyMap[key] = newNode
    }
    // is eviction necessary
    if len(l.keyMap) > l.capacity {
        nodeRemoved := l.list.RemoveTail()
        delete(l.keyMap, nodeRemoved.Data.key)
    }
}
Copier après la connexion

Pour Put(), nous vérifions d’abord s’il existe déjà une valeur pour la clé donnée. Si tel est le cas, mettez à jour la valeur et déplacez le nœud en tête de liste. Sinon, nous créons une nouvelle entrée de cache, l'ajoutons à la liste en tant qu'en-tête et l'ajoutons à notre carte.

Enfin, n'oubliez pas de vérifier la capacité. Si la nouvelle entrée nous dépasse la capacité, nous expulsons l'entrée la plus ancienne qui est la queue de la liste et supprimons l'entrée de notre carte.

Notez que stocker la clé dans le cadre de l'entrée du cache nous permet de supprimer rapidement la clé de la carte. Si nous avions uniquement stocké les données dans l’entrée du cache, nous aurions alors besoin de parcourir la carte pour les trouver.

Il manque quelque chose de critique dans ce cache pour une application multithread. Il n'y a pas de synchronisation. En réalité, un cache serait accessible par plusieurs threads. La synchronisation est un sujet complexe. Pour notre implémentation, nous pouvons ajouter un mutex à la structure du cache :

type lruCache[T any] struct {
    capacity int
    keyMap   map[string]DoubleNode[lruCacheEntry[T]]
    list     DoubleLinkedList[lruCacheEntry[T]]
    mutex    sync.RWMutex
}
Copier après la connexion

puis ajoutez ce qui suit au début de chaque fonction.

    l.mutex.Lock()
    defer l.mutex.Unlock()
Copier après la connexion

Remarquez que nous utilisons un verrou en lecture/écriture. Certaines fonctions ne changent pas la structure du cache, nous pouvons donc utiliser la méthode de verrouillage en lecture fournie, par exemple la fonction Len() :

func (l *lruCache[T]) Len() int {
    l.mutex.RLock()
    defer l.mutex.RUnlock()
    return len(l.keyMap)
}
Copier après la connexion

Notez que la stratégie de synchronisation choisie ici peut échouer si un grand nombre de threads tentent d'accéder au cache. C'est un sujet complexe qui pourrait faire l'objet d'une série de posts en soi.

Voir l'implémentation complète dans le référentiel indiqué dans le lien ci-dessous.

Que feriez-vous différemment pour implémenter un cache ? Comment aborderiez-vous la synchronisation ? Je suis intéressé à entendre vos réflexions sur celui-ci. Il n'y a pas de solution unique à ce problème, alors laissez vos commentaires ci-dessous.

Merci !

Le code de cet article et de tous les articles de cette série peut être trouvé ici

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!

Déclaration de ce site Web
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn

Outils d'IA chauds

Undresser.AI Undress

Undresser.AI Undress

Application basée sur l'IA pour créer des photos de nu réalistes

AI Clothes Remover

AI Clothes Remover

Outil d'IA en ligne pour supprimer les vêtements des photos.

Undress AI Tool

Undress AI Tool

Images de déshabillage gratuites

Clothoff.io

Clothoff.io

Dissolvant de vêtements AI

Video Face Swap

Video Face Swap

Échangez les visages dans n'importe quelle vidéo sans effort grâce à notre outil d'échange de visage AI entièrement gratuit !

Article chaud

<🎜>: Bubble Gum Simulator Infinity - Comment obtenir et utiliser les clés royales
4 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌
<🎜>: Grow A Garden - Guide de mutation complet
3 Il y a quelques semaines By DDD
Nordhold: Système de fusion, expliqué
4 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌
Mandragora: Whispers of the Witch Tree - Comment déverrouiller le grappin
3 Il y a quelques semaines By 尊渡假赌尊渡假赌尊渡假赌

Outils chauds

Bloc-notes++7.3.1

Bloc-notes++7.3.1

Éditeur de code facile à utiliser et gratuit

SublimeText3 version chinoise

SublimeText3 version chinoise

Version chinoise, très simple à utiliser

Envoyer Studio 13.0.1

Envoyer Studio 13.0.1

Puissant environnement de développement intégré PHP

Dreamweaver CS6

Dreamweaver CS6

Outils de développement Web visuel

SublimeText3 version Mac

SublimeText3 version Mac

Logiciel d'édition de code au niveau de Dieu (SublimeText3)

Sujets chauds

Tutoriel Java
1671
14
Tutoriel PHP
1276
29
Tutoriel C#
1256
24
Golang vs Python: performance et évolutivité Golang vs Python: performance et évolutivité Apr 19, 2025 am 12:18 AM

Golang est meilleur que Python en termes de performances et d'évolutivité. 1) Les caractéristiques de type compilation de Golang et le modèle de concurrence efficace le font bien fonctionner dans des scénarios de concurrence élevés. 2) Python, en tant que langue interprétée, s'exécute lentement, mais peut optimiser les performances via des outils tels que Cython.

Golang et C: concurrence vs vitesse brute Golang et C: concurrence vs vitesse brute Apr 21, 2025 am 12:16 AM

Golang est meilleur que C en concurrence, tandis que C est meilleur que Golang en vitesse brute. 1) Golang obtient une concurrence efficace par le goroutine et le canal, ce qui convient à la gestion d'un grand nombre de tâches simultanées. 2) C Grâce à l'optimisation du compilateur et à la bibliothèque standard, il offre des performances élevées près du matériel, adaptées aux applications qui nécessitent une optimisation extrême.

Partage avec Go: un guide du débutant Partage avec Go: un guide du débutant Apr 26, 2025 am 12:21 AM

GOISIDEALFORBEGINNERNERS et combinant pour pourcloudandNetWorkServicesDuetOtssimplicity, Efficiency, andCurrencyFeatures.1) InstallgofromTheofficialwebsiteandverifywith'goversion'..2)

Golang vs C: Performance et comparaison de la vitesse Golang vs C: Performance et comparaison de la vitesse Apr 21, 2025 am 12:13 AM

Golang convient au développement rapide et aux scénarios simultanés, et C convient aux scénarios où des performances extrêmes et un contrôle de bas niveau sont nécessaires. 1) Golang améliore les performances grâce à des mécanismes de collecte et de concurrence des ordures, et convient au développement de services Web à haute concurrence. 2) C réalise les performances ultimes grâce à la gestion manuelle de la mémoire et à l'optimisation du compilateur, et convient au développement du système intégré.

Impact de Golang: vitesse, efficacité et simplicité Impact de Golang: vitesse, efficacité et simplicité Apr 14, 2025 am 12:11 AM

GOIMIMPACTSDEVENCEMENTSPOSITIVEMENTS INSPECT, EFFICACTION ET APPLICATION.1) VITESSE: GOCOMPILESQUICKLYANDRUNSEFFIÉMENT, IDEALFORLARGEPROROSTS.2) Efficacité: ITSCOMPEHENSIVESTANDARDLIBRARYREDUCEEXTERNEDENDENCES, EnhancingDevelovefficiency.3) Simplicité: Simplicité: Implicité de la manière

Golang vs Python: différences et similitudes clés Golang vs Python: différences et similitudes clés Apr 17, 2025 am 12:15 AM

Golang et Python ont chacun leurs propres avantages: Golang convient aux performances élevées et à la programmation simultanée, tandis que Python convient à la science des données et au développement Web. Golang est connu pour son modèle de concurrence et ses performances efficaces, tandis que Python est connu pour sa syntaxe concise et son écosystème de bibliothèque riche.

Golang et C: les compromis en performance Golang et C: les compromis en performance Apr 17, 2025 am 12:18 AM

Les différences de performance entre Golang et C se reflètent principalement dans la gestion de la mémoire, l'optimisation de la compilation et l'efficacité du temps d'exécution. 1) Le mécanisme de collecte des ordures de Golang est pratique mais peut affecter les performances, 2) la gestion manuelle de C et l'optimisation du compilateur sont plus efficaces dans l'informatique récursive.

La course de performance: Golang vs C La course de performance: Golang vs C Apr 16, 2025 am 12:07 AM

Golang et C ont chacun leurs propres avantages dans les compétitions de performance: 1) Golang convient à une concurrence élevée et à un développement rapide, et 2) C fournit des performances plus élevées et un contrôle fin. La sélection doit être basée sur les exigences du projet et la pile de technologie d'équipe.

See all articles