Data Structure Doubly Linked Lists

Introduction to Doubly Linked List

A doubly linked list is a linear data structure that consists of a sequence of elements called nodes. Each node contains two pointers, one pointing to the previous node and the other pointing to the next node in the sequence. This allows for traversal in both directions: forward and backward. In addition to the usual operations of insertion and deletion, a doubly linked list also supports operations like insertion at the beginning and end, as well as traversal in both directions.

Doubly linked lists are often used in situations where efficient insertion and deletion operations are required, such as in implementing data structures like stacks, queues, and hash tables. The ability to traverse the list in both directions can be particularly useful in scenarios where accessing elements from the end of the list is a common operation.
One advantage of doubly linked lists over singly linked lists is that they provide more flexibility in terms of operations. For example, in a singly linked list, if we want to delete a node, we need to traverse the list to find the node that comes before it. However, in a doubly linked list, we can easily access the previous node using the backward pointer, making deletion operations more efficient.
Another advantage of doubly linked lists is that they allow for easy implementation of algorithms that require backward traversal, such as reversing the list or finding the kth element from the end. With a singly linked list, these operations would require additional steps and complexity.
However, there are also some trade-offs to consider when using doubly linked lists. One is the extra memory required to store the backward pointers, which increases the overall memory usage compared to singly linked lists. Additionally, the extra pointers also increase the complexity of some operations, such as inserting or deleting a node, as we need to update multiple pointers to maintain the integrity of the list.
In conclusion, doubly linked lists are a versatile data structure that provides efficient insertion and deletion operations, as well as the ability to traverse the list in both directions. They are commonly used in various applications and can be particularly useful in scenarios where backward traversal or frequent insertion and deletion operations are required. However, the trade-offs in terms of memory usage and complexity should be carefully considered when deciding whether to use a doubly linked list in a specific situation.

Structure of a Doubly Linked List

A typical node in a doubly linked list contains three components:

  1. Data: The actual value or data that the node holds.
  2. Previous: A pointer to the previous node in the sequence.
  3. Next: A pointer to the next node in the sequence.

The first node of the doubly linked list is called the head, and the last node is called the tail. The head’s previous pointer is always null, indicating the start of the list, while the tail’s next pointer is null, indicating the end of the list.

A doubly linked list is a data structure that consists of a sequence of nodes, where each node contains a value and two pointers. Unlike a singly linked list, which only has a pointer to the next node, a doubly linked list has two pointers: one pointing to the previous node and one pointing to the next node.

This additional pointer to the previous node allows for more efficient traversal of the list in both directions. It enables operations such as inserting a new node before a given node or deleting a node from the list without having to iterate through the entire list.

The structure of a doubly linked list is simple yet powerful. Each node contains the actual data it holds, as well as pointers to the previous and next nodes in the sequence. These pointers allow for easy navigation through the list, whether forwards or backwards.

The head of the list is the first node, and its previous pointer is always null since there is no node before it. Similarly, the tail of the list is the last node, and its next pointer is null because there is no node after it.

By utilizing the structure of a doubly linked list, various operations can be performed efficiently. For example, inserting a new node at the beginning of the list involves updating the pointers of the new node, the current head, and the previous head. Similarly, deleting a node requires updating the pointers of the previous and next nodes to bypass the node being deleted.

In summary, the structure of a doubly linked list consists of nodes that hold data and have pointers to the previous and next nodes. This structure allows for efficient traversal and manipulation of the list, making it a valuable data structure in many applications.

Let’s continue with the example of creating a doubly linked list:

Node 1: {Data: 10, Previous: null, Next: Node 2}
Node 2: {Data: 20, Previous: Node 1, Next: Node 3}
Node 3: {Data: 30, Previous: Node 2, Next: Node 4}
Node 4: {Data: 40, Previous: Node 3, Next: null}

In this updated state, we have added a fourth node with data value 40. The previous pointer of Node 4 points to Node 3, and the next pointer is null, indicating the end of the list.

Node 1: {Data: 10, Previous: null, Next: Node 2}
Node 2: {Data: 20, Previous: Node 1, Next: Node 3}
Node 3: {Data: 30, Previous: Node 2, Next: Node 4}
Node 4: {Data: 40, Previous: Node 3, Next: Node 5}
Node 5: {Data: 50, Previous: Node 4, Next: null}

Continuing further, we add a fifth node with data value 50. The previous pointer of Node 5 points to Node 4, and the next pointer is null, indicating the end of the list.

Node 1: {Data: 10, Previous: null, Next: Node 2}
Node 2: {Data: 20, Previous: Node 1, Next: Node 3}
Node 3: {Data: 30, Previous: Node 2, Next: Node 4}
Node 4: {Data: 40, Previous: Node 3, Next: Node 5}
Node 5: {Data: 50, Previous: Node 4, Next: Node 6}
Node 6: {Data: 60, Previous: Node 5, Next: null}

Lastly, we add a sixth node with data value 60. The previous pointer of Node 6 points to Node 5, and the next pointer is null, indicating the end of the list.

By following this process, we can continue to add more nodes to the doubly linked list, creating a chain of interconnected nodes where each node holds a data value and references the previous and next nodes in the list.

Advantages of Doubly Linked List

Doubly linked lists have several advantages over other data structures:

  1. Bi-directional traversal: Unlike singly linked lists, doubly linked lists allow for traversal in both directions. This makes it easier to navigate through the list and perform operations like searching, insertion, and deletion.
  2. Insertion and deletion at both ends: Doubly linked lists support efficient insertion and deletion at both the beginning and end of the list. This is particularly useful in scenarios where frequent modifications are required.
  3. Efficient reverse traversal: As each node has a pointer to the previous node, reverse traversal of a doubly linked list can be performed efficiently. This is useful in scenarios where accessing elements in reverse order is required.
  4. Efficient implementation of algorithms: Doubly linked lists are often utilized in various algorithms due to their advantages. For example, in sorting algorithms like QuickSort and MergeSort, doubly linked lists can be used to efficiently partition and merge elements.
  5. Undo/Redo functionality: Doubly linked lists are commonly used in applications that require undo and redo functionality. Each operation can be represented as a node in the list, and the ability to traverse both forward and backward allows for easy undoing and redoing of operations.
  6. Efficient implementation of data structures: Doubly linked lists are a fundamental building block for many other data structures. For instance, they are used in the implementation of stacks and queues, where efficient insertion and deletion at both ends are required.
  7. Circular doubly linked lists: Doubly linked lists can be circular, where the last node points back to the first node. This allows for efficient traversal and looping through the elements of the list.
  8. Memory efficiency: While doubly linked lists require more memory compared to singly linked lists due to the extra pointer for the previous node, they are still more memory-efficient than other data structures like arrays. This is because doubly linked lists can dynamically allocate and deallocate memory as needed.

3. Searching

Searching for a specific value in a doubly linked list involves traversing through the list and comparing the data of each node with the desired value. The following steps are performed:

  1. Start from the head of the list.
  2. Compare the data of the current node with the desired value.
  3. If a match is found, return the node.
  4. If the end of the list is reached without finding a match, return null to indicate that the value was not found.

Searching in a doubly linked list can be more efficient than in a singly linked list because the list can be traversed in both forward and backward directions.

4. Traversing

Traversing a doubly linked list involves visiting each node in the list in a specific order. The following steps are performed:

  1. Start from the head of the list.
  2. Visit the data of the current node.
  3. Move to the next node by following the next pointer.
  4. Repeat steps 2-3 until the end of the list is reached (i.e., the next pointer of the current node is null).

Traversing a doubly linked list can be done in both forward and backward directions, allowing for more flexibility in accessing and processing the data.

5. Updating

Updating the data of a node in a doubly linked list involves finding the desired node and modifying its data. The following steps are performed:

  1. Find the node with the desired data.
  2. Update the data of the found node with the new value.

Updating a node in a doubly linked list can be done more efficiently than in a singly linked list because the previous pointer allows direct access to the previous node.

Example: Operations on Doubly Linked List

Let’s consider an example to understand how operations are performed on a doubly linked list:

Initial Doubly Linked List: None

In this initial state, the doubly linked list is empty.

Insert 10 at the Beginning:
Doubly Linked List: 10

We start by inserting a node with a data value of 10 at the beginning of the doubly linked list. This operation results in a doubly linked list with a single node containing the value 10.

Insert 20 at the End:
Doubly Linked List: 10 <-> 20

Next, we insert a node with a data value of 20 at the end of the doubly linked list. This operation adds a new node to the end of the list and updates the previous node’s next pointer to point to the new node. The new node’s previous pointer is set to the previous last node in the list.

Insert 30 at Position 2:
Doubly Linked List: 10 <-> 30 <-> 20

Now, we insert a node with a data value of 30 at position 2 in the doubly linked list. This operation involves updating the next and previous pointers of the neighboring nodes to include the new node. The new node’s next pointer is set to the node that was previously at position 2, and its previous pointer is set to the node that was previously at position 1.

Delete from the Beginning:
Doubly Linked List: 30 <-> 20

After that, we perform a deletion operation from the beginning of the doubly linked list. This operation removes the first node in the list and updates the next pointer of the previous first node to point to the new first node. The new first node’s previous pointer is set to null.

Delete from the End:
Doubly Linked List: 30

Next, we delete a node from the end of the doubly linked list. This operation removes the last node in the list and updates the previous node’s next pointer to null. The new last node’s previous pointer is set to the previous second-to-last node in the list.

Delete from Position 1:
Doubly Linked List: None

Finally, we perform a deletion operation from position 1 in the doubly linked list. This operation removes the only node in the list and sets the doubly linked list to an empty state.

In this example, we have demonstrated the various operations that can be performed on a doubly linked list, including insertion and deletion at the beginning, end, and specific positions. These operations allow for flexibility in manipulating the data stored in the doubly linked list.

Scroll to Top