Erzählen Sie mir von verknüpften Listen
Nachdem ich kürzlich ein Coding-Bootcamp absolviert habe, habe ich mich intensiv mit Datenstrukturen und Algorithmen befasst, um mein Grundwissen zu verbessern. Verknüpfte Listen sind eine relativ einfache Datenstruktur, bilden jedoch die Grundlage für komplexere Datenstrukturen. In diesem Artikel werden verknüpfte Listen behandelt - ihre Stärken, Schwächen und deren Implementierung.
Datenstrukturen sind einfach ausgedrückt Möglichkeiten, Daten effizient zu organisieren und zu speichern
Was sind Datenstrukturen?
Datenstrukturen sind einfach ausgedrückt Möglichkeiten, Daten effizient zu organisieren und zu speichern. Diese ermöglichen es uns, Informationen zu speichern, damit wir später darauf zugreifen können - nach einem bestimmten Artikel suchen oder sie nach einer bestimmten Reihenfolge sortieren oder sogar große Datenmengen verwalten. JavaScript verfügt über Datenstrukturen, mit denen Sie wahrscheinlich sehr vertraut sind - Arrays und Objekte. Dies sind primitive Datenstrukturen, die der Sprache eigen sind. Nicht-primitive Datenstrukturen sind Datenstrukturen, die vom Programmierer definiert werden. Beispiele hierfür sind Stapel, Warteschlangen, Bäume, Diagramme und Hash-Tabellen.
Was sind verknüpfte Listen?
Eine verknüpfte Liste ist eine lineare Datenstruktur, die Elemente in sequentieller Reihenfolge enthält. Klingt bekannt? Sie ähneln Arrays, sind aber nicht ganz gleich. Arrays sind nullindiziert ist , jeden Index Bedeutung wird abgebildet auf einen Wert, und das erste Element des Arrays beginnt bei einem Index von 0. Sie können ganz einfach Elemente aus dem Array ziehen, so lange , wie Sie den Index kennen.
Andererseits besteht eine verknüpfte Liste aus Knoten, die auf andere Knoten verweisen . Jeder Knoten kennt nur sich selbst und den nächsten Knoten. Als lineare Datenstruktur sind die Elemente der Reihe nach angeordnet. Man könnte es sich als Zug vorstellen - jeder Triebwagen ist mit dem nächsten Triebwagen verbunden und so weiter. Verknüpfte Listen sind nicht indiziert , sodass Sie nicht direkt auf das 5. Element zugreifen können. Sie müssten am Anfang beginnen, zum nächsten gehen und zum nächsten gehen, bis Sie das 5. Element finden. Der erste Knoten heißt Kopf und der letzte Knoten heißt Schwanz. Es gibt verschiedene Arten von verknüpften Listen - einfach verknüpfte Listen, doppelt verknüpfte Listen und zirkuläre verknüpfte Listen. Für diesen Artikel werde ich mich auf die einfach verknüpfte Liste konzentrieren.
Der Zugriff auf Elemente im gesamten Array ist einfacher, dies ist jedoch mit Kosten verbunden.
Warum verknüpfte Listen über Arrays verwenden?
Sie fragen sich vielleicht - aber warum sollten Sie dies verwenden, wenn ich mit Arrays direkt auf Elemente zugreifen kann? Das ist wahr! Der Zugriff auf Elemente im gesamten Array ist einfacher, dies ist jedoch mit Kosten verbunden.
Arrays werden indiziert. Wenn Sie also ein Element am Anfang oder in der Mitte entfernen müssen, müssen alle folgenden Indizes verschoben werden. Dies gilt auch, wenn Sie ein Element einfügen müssen - alle folgenden Elemente müssen neu indiziert werden. Wenn Sie Elemente in eine verknüpfte Liste einfügen oder daraus entfernen, müssen Sie sie nicht neu indizieren - alles, was Sie interessiert, ist der Knoten, auf den sie zeigt .
Implementieren einer einfach verknüpften Liste
Es gibt verschiedene Arten von verknüpften Listen - einzeln, doppelt und zirkulär. Einfach verknüpfte Listen sind eine Art verknüpfter Listen, bei denen jeder Knoten einen Wert und einen Zeiger auf den nächsten Knoten hat (null, wenn es sich um den Endknoten handelt). Doppelt verknüpfte Listen haben dagegen einen zusätzlichen Zeiger auf den vorherigen Knoten. In kreisförmig verknüpften Listen zeigt der Endknoten zurück zum Kopfknoten.
Hinweis: Wenn Sie visuell lernen, ist VisuAlgo ein großartiger Ort, um herumzuspielen und zu sehen, wie sich die Datenstruktur ändert.
Wenn Sie den vollständigen Code überspringen möchten, lesen Sie diese GitHub-Übersicht .
Erstellen eines Knotens
Beginnen wir mit dem Aufbau eines zu verwendenden Knotens. Es hat einen Wert und einen nächsten Zeiger, der den nachfolgenden Nachbarn anzeigt.
class Node {
constructor(val) {
this.val = val;
this.next = null;
}
}
Wir beginnen mit dem Aufbau einer Klasse namens SinglyLinkedList und geben ihr eine Grundlage für den Kopf, den Schwanz und die Länge der Liste.
class SinglyLinkedList {
constructor() {
this.head = null;
this.tail = null;
this.length = 0;
}
}
Um am Ende einen Knoten hinzuzufügen, prüfen wir, ob in der Liste bereits ein Kopf deklariert ist. Wenn nicht, setzen Sie Kopf und Schwanz als neu erstellten Knoten. Wenn bereits eine head-Eigenschaft vorhanden ist, setzen Sie die nächste Eigenschaft am Ende auf den neuen Knoten und die Eigenschaft tail auf den neuen Knoten. Wir erhöhen die Länge um 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;
}
Um einen Knoten am Ende zu entfernen, müssen wir zuerst prüfen, ob die Liste Knoten enthält. Wenn es Knoten gibt, müssen wir die Liste durchlaufen, bis wir den Schwanz erreichen. Wir setzen die nächste Eigenschaft des vorletzten Knotens auf null, setzen die Eigenschaft tail auf diesen Knoten, verringern die Länge um 1 und geben den Wert des entfernten Knotens zurück.
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;
}
Um den Knoten von Anfang an zu entfernen, prüfen wir, ob die Liste Knoten enthält. Wir speichern die aktuelle head-Eigenschaft in einer Variablen und setzen den head als nächsten Knoten des aktuellen Heads. Verringern Sie die Länge um 1 und geben Sie den Wert des entfernten Knotens zurück.
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
}
Um am Anfang einen Knoten hinzuzufügen, erstellen wir zuerst den Knoten und prüfen, ob die Liste eine head-Eigenschaft enthält. Wenn ja, setzen Sie die nächste Eigenschaft des neu erstellten Knotens auf den aktuellen Kopf, setzen Sie den Kopf auf den neuen Knoten und erhöhen Sie die Länge.
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
}
Um auf einen Knoten basierend auf seiner Position in der Liste zuzugreifen, prüfen wir zunächst, ob der Index ein gültiger Index ist. Danach durchlaufen wir die Liste, während wir eine Zählervariable beibehalten. Sobald die Zählervariable mit dem Index übereinstimmt, geben wir den gefundenen Knoten zurück
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;
}
Um einen Knoten basierend auf seiner Position zu ändern, suchen wir zuerst den Knoten mithilfe der get-Methode und setzen den Wert auf den neuen Wert.
set(idx, val) {
var foundNode = this.get(idx)
if (foundNode) {
foundNode.val = val
return true;
}
return false;
}
Um einen Knoten einzufügen, überprüfen wir zuerst den Index. Wenn es gleich 0 ist, verwenden wir unsere Methode zum Verschieben. Wenn es gleich der Länge ist, schieben wir es auf die Liste. Andernfalls erhalten wir den Vorher-Knoten, indem wir die get-Methode um eins unter dem Index verwenden. Im Wesentlichen müssen wir den Knoten vor und nach dem ausgewählten Index verfolgen, um sicherzustellen, dass unsere Zeiger auf den richtigen Knoten gerichtet sind.
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;
}
Um einen Knoten basierend auf der Position zu entfernen, verwenden wir erneut unsere get-Methode und verwenden dasselbe Prinzip, das wir zum Einfügen eines Knotens verwendet haben. Indem wir den Knoten vor dem ausgewählten Index verfolgen, können wir unsere Zeiger manövrieren.
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;
}
Stärken verknüpfter Listen
Der Vorteil der Verwendung einer verknüpften Liste besteht in der Möglichkeit, Elemente schnell einzufügen und zu entfernen. Da verknüpfte Listen nicht indiziert sind, ist das Einfügen von Elementen relativ schnell und einfach.
Zum Hinzufügen am Ende der verknüpften Liste muss lediglich der alte Schwanz auf den neuen Knoten zeigen und die Eigenschaft tail auf den neuen Knoten gesetzt werden. Das Hinzufügen am Anfang der verknüpften Liste ist dieselbe Voraussetzung, aber der neue Knoten zeigt auf den alten Kopf, und die Eigenschaft head wird auf den neuen Knoten festgelegt. Grundsätzlich gleiche Aktionen, also ist die Einfügung O (1).
Hinweis: Wenn Sie in die Mitte einer verknüpften Liste einfügen, müssen Sie nach Ihrem Knoten suchen (O (N)) und dann den Knoten einfügen (O (1)).
Das Entfernen eines Knotens aus der verknüpften Liste kann einfach sein… aber auch nicht. Das hängt davon ab, wo. Das Entfernen eines Knotens vom Anfang der Liste ist einfach - der zweite Knoten wird zum neuen Kopf, und wir setzen den nächsten Zeiger des alten Kopfes auf Null. Das Entfernen vom Ende ist schwieriger, da hierfür die gesamte Liste durchgesehen und am vorletzten Knoten angehalten werden muss. Worst-Case O (N) und Best-Case O (1).
Schwächen verknüpfter Listen
Schwächen einer verknüpften Liste werden durch die fehlende Neuindizierung verursacht. Da Sie nicht direkt auf Elemente zugreifen können, wird das Suchen und Zugreifen auf Elemente erschwert.
Das Suchen und Zugreifen auf einen Knoten in einer verknüpften Liste erfordert, dass Sie am Anfang der Liste beginnen und den Spuren der Brotkrumen in der Liste folgen. Wenn wir am Ende der Liste nach dem Knoten suchen, müssen wir die gesamte verknüpfte Liste durchgehen, wodurch die Zeitkomplexität effektiv O (N) wird. Wenn die Liste wächst, wächst die Anzahl der Operationen. Im Vergleich zu einem Array mit wahlfreiem Zugriff dauert der Zugriff auf ein Element konstant lange.
Das ist es!
Verknüpfte Listen sind großartige Alternativen zu Arrays, wenn das Einfügen und Löschen zu Beginn Schlüsselaktionen sind. Arrays werden indiziert, verknüpfte Listen hingegen nicht. Verknüpfte Listen sind auch die grundlegende Datenstruktur, aus der Stapel und Warteschlangen aufgebaut sind, auf die ich in einem späteren Artikel eingehen werde. Wenn Sie es bis hierher geschafft haben, ein großes Lob an Sie und ich hoffe, Sie haben zusammen mit Ihrer Programmierreise ein kleines Nugget gelernt!