Alors, parlez-moi des listes liées

Sep 21 2020
Ayant récemment obtenu mon diplôme d'un bootcamp de codage, je me suis plongé dans l'étude des structures de données et des algorithmes pour améliorer mes connaissances fondamentales. Les listes liées sont une structure de données relativement simple, mais elles constituent la base de structures de données plus complexes à l'avenir.

Ayant récemment obtenu mon diplôme d'un bootcamp de codage, je me suis plongé dans l'étude des structures de données et des algorithmes pour améliorer mes connaissances fondamentales. Les listes liées sont une structure de données relativement simple, mais elles constituent la base de structures de données plus complexes à l'avenir. Cet article abordera les listes chaînées - leurs forces, leurs faiblesses et la manière de les mettre en œuvre.

Les structures de données sont, en termes simples, des moyens d'organiser et de conserver efficacement les données

Que sont les structures de données?

Les structures de données sont, en termes simples, des moyens d'organiser et de conserver efficacement les données. Ceux-ci nous permettent de conserver des informations afin que nous puissions y accéder pour plus tard - en recherchant un élément spécifique, ou en les triant pour avoir un certain ordre ou même en gérant de grands ensembles de données. JavaScript a des structures de données avec lesquelles vous êtes probablement très familier - des tableaux et des objets. Ce sont des structures de données primitives natives du langage. Les structures de données non primitives sont des structures de données définies par le programmeur. Les exemples incluent les piles, les files d'attente, les arbres, les graphiques et les tables de hachage.

Que sont les listes liées?

Une liste chaînée est une structure de données linéaire qui contient les éléments dans un ordre séquentiel. Semble familier? Ils sont similaires aux tableaux, mais ils ne sont pas tout à fait les mêmes. Les tableaux sont indexés à zéro , ce qui signifie que chaque index est mappé à une valeur, et le premier élément du tableau commence à un index de 0. Vous pouvez facilement extraire des éléments du tableau, tant que vous connaissez l'index.

D'autre part, une liste liée se compose de nœuds qui pointent vers d'autres nœuds. Chaque nœud ne connaît que lui-même et le nœud suivant. En tant que structure de données linéaire, les éléments sont organisés dans l'ordre. Vous pouvez le considérer comme un train - chaque wagon est connecté au wagon suivant, et ainsi de suite. Les listes liées ne sont pas indexées , vous ne pouvez donc pas accéder directement au 5ème élément. Vous devrez commencer par le début, passer au suivant et passer au suivant jusqu'à ce que vous trouviez le 5e élément. Le premier nœud est appelé la tête et le dernier nœud est appelé la queue. Il existe différents types de listes chaînées: liste chaînée simple, liste double chaînée et liste chaînée circulaire. Pour le bien de cet article, je vais me concentrer sur la liste liée individuellement.

L'accès aux éléments dans tout le tableau est plus facile, mais cela a un coût.

Pourquoi utiliser des listes liées sur des tableaux?

Vous vous demandez peut-être - mais pourquoi l'utiliser si je peux accéder directement aux éléments avec des tableaux? C'est vrai! L'accès aux éléments dans tout le tableau est plus facile, mais cela a un coût.

Les tableaux sont indexés - donc chaque fois que vous devez supprimer un élément du début ou du milieu, tous les index suivants doivent être décalés. Cela est également vrai si vous devez insérer un élément - tous les éléments suivants doivent être réindexés. Lorsque vous insérez ou supprimez des éléments d'une liste chaînée, vous n'avez pas besoin de réindexer - tout ce qui compte, c'est le nœud vers lequel il pointe .

Implémentation d'une liste liée individuellement

Il existe différents types de listes chaînées - simples, doubles et circulaires. Les listes liées simples sont un type de liste chaînée où chaque nœud a une valeur et un pointeur vers le nœud suivant (nul s'il s'agit du nœud de queue). Les listes doublement liées, en revanche, ont un pointeur supplémentaire vers le nœud précédent. Les listes liées circulaires sont l'endroit où le nœud de queue pointe vers le nœud de tête.

Remarque: Si vous êtes un apprenant visuel, VisuAlgo est un excellent endroit pour jouer et voir la structure des données subir des changements.

Si vous souhaitez ignorer le code complet: consultez ce GitHub gist .

Créer un nœud

Commençons par construire un nœud à utiliser. Il aura une valeur et un pointeur suivant, qui indiquera son voisin suivant.

class Node {    
    constructor(val) {        
        this.val = val;        
        this.next = null;    
    }
}

Nous allons commencer par construire une classe appelée SinglyLinkedList et lui donner une base pour la tête, la queue et la longueur de la liste.

class SinglyLinkedList {    
      constructor() {        
          this.head = null;
          this.tail = null;
          this.length = 0;
      }
}

Pour ajouter un nœud à la fin, nous vérifierons s'il y a une tête déjà déclarée sur la liste. Sinon, définissez la tête et la queue comme le nœud nouvellement créé. S'il existe déjà une propriété head, définissez la propriété suivante de la queue sur le nouveau nœud et la propriété tail sur le nouveau nœud. Nous incrémenterons la longueur de 1.

push(val) {
        var newNode = new Node(val)
        if (!this.head) {
            this.head = newNode
            this.tail = newNode
        } else {
            this.tail.next = newNode
            this.tail = newNode
        }
        this.length++;
        return this;
}

Pour supprimer un nœud à la fin, nous devons d'abord vérifier s'il y a des nœuds dans la liste. S'il y a des nœuds, nous devons parcourir la liste jusqu'à ce que nous atteignions la queue. Nous allons définir la propriété suivante de l'avant-dernier nœud sur null, définir la propriété tail sur ce nœud, décrémenter la longueur de 1 et renvoyer la valeur du nœud supprimé.

Suppression du nœud de queue, nœud 77.

pop() {
        if (!this.head) return undefined;
        var current = this.head
        var newTail = current
        while (current.next) {
            newTail = current
            current = current.next;
        }
        this.tail = newTail;
        this.tail.next = null;
        this.length--;
        if (this.length === 0) {
            this.head = null;
            this.tail = null;
        }
        return current;
}

Pour supprimer le nœud du début, nous vérifions s'il y a des nœuds dans la liste. Nous allons stocker la propriété head actuelle dans une variable et définir la tête comme étant le nœud suivant de la tête actuelle. Décrémentez la longueur de 1 et renvoyez la valeur du nœud supprimé.

shift() {
        if (!this.head) return undefined
        var oldHead = this.head;
        this.head = oldHead.next;
        this.length--
        if (this.length===0) {
            this.head = null;
            this.tail = null;
        }
        return oldHead
}

Pour ajouter un nœud au début, nous créons d'abord le nœud et vérifions s'il y a une propriété head sur la liste. Si tel est le cas, définissez la propriété suivante du nœud nouvellement créé sur la tête actuelle, définissez head sur le nouveau nœud et incrémentez la longueur.

Ajoutez le nœud 61 au début de la liste.

unshift(val) {
        var newNode = new Node(val)
        if (!this.head) {
            this.head = newNode;
            this.tail = newNode;
        } else {
            newNode.next = this.head;
            this.head = newNode;
        }
        this.length++
        return this
}

Pour accéder à un nœud en fonction de sa position dans la liste, nous allons d'abord vérifier si l'index est un index valide. Ensuite, nous allons parcourir la liste, tout en gardant une variable de compteur. Une fois que la variable de compteur correspond à l'index, nous retournerons le nœud trouvé

get(idx) {
        if (idx < 0 || idx >= this.length) return null
        var counter = 0;
        var current = this.head
        while (counter !== idx) {
            current = current.next;
            counter++;
        }
        return current;
}

Pour changer un nœud en fonction de sa position, nous trouvons d'abord le nœud, en utilisant la méthode get et définissons la valeur sur la nouvelle valeur.

set(idx, val) {
        var foundNode = this.get(idx)
        if (foundNode) {
            foundNode.val = val
            return true;
        }
        return false;
}

Pour insérer un nœud, nous vérifions d'abord l'index. S'il est égal à 0, nous utilisons notre méthode unshift. S'il est égal à la longueur, nous le plaçons dans la liste. Sinon, nous obtiendrons le nœud avant en utilisant la méthode get à un de moins que l'index. Essentiellement, nous devons garder une trace du nœud avant et après l'index sélectionné pour nous assurer que nos pointeurs sont dirigés vers le nœud correct.

Insertion du nœud 60 à la position 3 de la liste.

insert(idx, val) {
        if (idx < 0 || idx > this.length) return false;
        if (idx === 0) {
            this.unshift(val)
            return true;
        };
        if (idx === this.length) {
            this.push(val);
            return true;
        };
        var newNode = new Node(val);
        var beforeNode = this.get(idx-1);
        var afterNode = beforeNode.next;
        beforeNode.next = newNode;
        newNode.next = afterNode;
        this.length++;
        return true;
}

Pour supprimer un nœud en fonction de la position, nous utiliserons à nouveau notre méthode get et utiliserons le même principe que celui utilisé pour insérer un nœud. En gardant une trace du nœud avant l'index choisi, nous pouvons manœuvrer nos pointeurs.

remove(idx) {
        if (idx < 0 || idx > this.length) return undefined;
        if (idx === this.length-1) {
            this.pop()
        }
        if (idx === 0) {
            this.shift()
        }
        var prevNode = this.get(idx-1)
        var removeNode = prevNode.next
        prevNode.next = removeNode.next
        this.length--;
        return removeNode;
}

Points forts des listes liées

L'avantage d'utiliser une liste chaînée est la possibilité d'insérer et de supprimer rapidement des éléments. Étant donné que les listes liées ne sont pas indexées, l'insertion d'éléments est relativement rapide et facile.

L'ajout à la fin de la liste liée nécessite simplement que l'ancienne queue pointe vers le nouveau nœud et que la propriété tail soit définie sur le nouveau nœud. L'ajout au début de la liste chaînée est le même principe, mais le nouveau nœud pointe vers l'ancienne tête et la propriété head est définie sur le nouveau nœud. Fondamentalement les mêmes actions, donc l'insertion est O (1).

Remarque: l'insertion au milieu d'une liste chaînée nécessite une recherche pour trouver votre nœud (O (N)), puis l'insertion du nœud (O (1)).

Supprimer un nœud de la liste chaînée peut être facile… mais pas non plus. Cela dépend d'où. Supprimer un nœud du début de la liste est simple - le deuxième nœud devient la nouvelle tête, et nous définissons le pointeur suivant de l'ancienne tête sur null. La suppression de la fin est plus délicate car cela nécessite de parcourir toute la liste et de s'arrêter à l'avant-dernier nœud. Le pire des cas O (N) et le meilleur des cas O (1).

Faiblesses des listes liées

Les faiblesses d'une liste chaînée sont causées par son manque de réindexation. Comme vous ne pouvez pas accéder directement aux éléments, cela rend la recherche et l'accès aux éléments plus difficiles.

La recherche et l'accès à un nœud dans une liste liée nécessite de commencer au début de la liste et de suivre la trace du fil d'Ariane dans la liste. Si nous recherchons le nœud à la fin de la liste, nous devrons parcourir toute la liste chaînée, ce qui rendra effectivement la complexité temporelle O (N). À mesure que la liste s'allonge, le nombre d'opérations augmente. Comparé à un tableau qui a un accès aléatoire, il faut un temps constant pour accéder à un élément.

C'est ça!

Les listes liées sont d'excellentes alternatives aux tableaux lorsque l'insertion et la suppression au début sont des actions clés. Les tableaux sont indexés, contrairement aux listes chaînées. Les listes liées sont également la structure de données fondamentale à partir de laquelle les piles et les files d'attente sont construites, ce dont je parlerai dans un article ultérieur. Si vous avez fait tout le chemin ici, bravo à vous et j'espère que vous avez appris une petite pépite avec votre parcours de programmation!

Références

Structures de données en JavaScript Qu'est-ce qu'une liste liée, de toute façon? [Partie 1]