How to Implement DoublyLinkedList in TypeScript

DoublyLinkedList is a data structure that contains a head, tail and length property. Linked Lists consist of nodes, and each node has a value and a pointer to another node or null.

The DoublyLinkedList is an extension of LinkedList. It has two pointers, one to the first element of the list and another to the last element of the list.

Following are some methods used in DoublyLinkedList:

InsertAtHead(): Inserts an element at the beginning of a DoublyLinkedList.

 insertAtHead(data: number) {
        const newNode = new Node(data);
        if (this.head === null) {
            this.head = newNode;
            this.tail = newNode;
        } else {
            this.head.prev = newNode;
            newNode.next = this.head;
            this.head = newNode;
        }
        this.length++;
    }
    ```

`InsertAtTail()`: Inserts an element at the end of a `DoublyLinkedList`.

```js
 insertAtTail(data: number) {
        const newNode = new Node(data);
        if (this.head === null) {
            this.head = newNode;
            this.tail = newNode;
        } else {
            newNode.prev = this.tail;
            this.tail.next = newNode;
            this.tail = newNode;
        }
        this.length++;
    }

InsertAtIndex(): Inserts an element at index n in a DoublyLinkedList.

 insertAtIndex(data: number, index: number) {
        if (index === 0) {
            this.insertAtHead(data);
            return;
        }
        if (index >= this.length) {
            this.insertAtTail(data);
            return;
        }
        const newNode = new Node(data);
        let currentNode = this.head;
        let currentIndex = 0;
        while (currentIndex !== index) {
            currentNode = currentNode.next;
            currentIndex++;
        }
        newNode.next = currentNode;
        newNode.prev = currentNode.prev;
        currentNode.prev.next = newNode;
        currentNode.prev = newNode;
        this.length++;
       
    }

DeleteAtHead(): Removes and returns the head node of this DoublyLinkedList.

 deleteAtHead() {
        if (this.head === null) {
            return null;
        }
        if (this.head === this.tail) {
            this.head = null;
            this.tail = null;
            this.length--;
            return;
        }
        this.head = this.head.next;
        this.head.prev = null;
        this.length--;
    }

DeleteAtTail(): Removes and returns the tail node of this DoublyLinkedList.

  deleteAtTail() {
        if (this.head === null) {
            return null;
        }
        if (this.head === this.tail) {
            this.head = null;
            this.tail = null;
            this.length--;
            return;
        }
        this.tail = this.tail.prev;
        this.tail.next = null;
        this.length--;
    }

Search(): Searches for a specified value within this DoublyLinkedList and returns its index if found; otherwise, returns -1

  search(data: number) {
        let currentNode = this.head;
        while (currentNode !== null) {
            if (currentNode.data === data) {
                return currentNode;
            }
            currentNode = currentNode.next;
        }
        return null;
    }

printList: This function prints the

   printList(){
        let currentNode = this.head;
        let list = [];
        while (currentNode !== null) {
            list.push(currentNode.data);
            currentNode = currentNode.next;
        }
        return list;
    }

Time Complexity

  • insertAtHead: O(1)
  • insertAtTail: O(1)
  • insertAtIndex: O(n)
  • deleteAtHead: O(1)
  • deleteAtTail: O(1)
  • deleteAtIndex: O(n)
  • search: O(n)
  • printList: O(n)

Complete Source Code Of DoublyLinkedList

Below is the complete source code



class Node {
    data: number;
    next: Node;
    prev: Node;
    constructor(data: number) {
        this.data = data;
        this.next = null;
        this.prev = null;
    }
    }

class DoublyLinkedList {
    head: Node;
    tail: Node;
    length: number;
    constructor() {
        this.head = null;
        this.tail = null;
        this.length = 0;
    }
    insertAtHead(data: number) {
        const newNode = new Node(data);
        if (this.head === null) {
            this.head = newNode;
            this.tail = newNode;
        } else {
            this.head.prev = newNode;
            newNode.next = this.head;
            this.head = newNode;
        }
        this.length++;
    }
    insertAtTail(data: number) {
        const newNode = new Node(data);
        if (this.head === null) {
            this.head = newNode;
            this.tail = newNode;
        } else {
            newNode.prev = this.tail;
            this.tail.next = newNode;
            this.tail = newNode;
        }
        this.length++;
    }
    insertAtIndex(data: number, index: number) {
        if (index === 0) {
            this.insertAtHead(data);
            return;
        }
        if (index >= this.length) {
            this.insertAtTail(data);
            return;
        }
        const newNode = new Node(data);
        let currentNode = this.head;
        let currentIndex = 0;
        while (currentIndex !== index) {
            currentNode = currentNode.next;
            currentIndex++;
        }
        newNode.next = currentNode;
        newNode.prev = currentNode.prev;
        currentNode.prev.next = newNode;
        currentNode.prev = newNode;
        this.length++;
       
    }
   
    deleteAtTail() {
        if (this.head === null) {
            return null;
        }
        if (this.head === this.tail) {
            this.head = null;
            this.tail = null;
            this.length--;
            return;
        }
        this.tail = this.tail.prev;
        this.tail.next = null;
        this.length--;
    }
   search(data: number) {
        let currentNode = this.head;
        while (currentNode !== null) {
            if (currentNode.data === data) {
                return currentNode;
            }
            currentNode = currentNode.next;
        }
        return null;
    }
    printList(){
        let currentNode = this.head;
        let list = [];
        while (currentNode !== null) {
            list.push(currentNode.data);
            currentNode = currentNode.next;
        }
        return list;
    }
}

const myDoublyLinkedList = new DoublyLinkedList();
myDoublyLinkedList.insertAtHead(10);
myDoublyLinkedList.insertAtHead(20);
myDoublyLinkedList.insertAtHead(30);
myDoublyLinkedList.insertAtTail(40);
myDoublyLinkedList.insertAtTail(50);
myDoublyLinkedList.insertAtTail(60);
myDoublyLinkedList.insertAtIndex(70, 3);
myDoublyLinkedList.deleteAtTail();
console.log(myDoublyLinkedList.printList()); // Output: [30, 20, 10, 70, 40, 50]

Next Post Previous Post
No Comment
Add Comment
comment url