Linked Lists
December 12, 2024 | Data Structures
Linked lists are one of the fundamental data structures in computer science. Unlike arrays, linked lists provide flexibility in memory allocation and ease of insertion and deletion operations. They form the backbone of many advanced data structures like stacks, queues, and graphs.
What is a Linked List?
A linked list is a linear data structure where each element, called a node, contains two parts:
- Data: The actual value or content.
- Pointer (or Reference): A link to the next node in the sequence.
Example of a Node in Python:
class Node:
def __init__(self, data):
self.data = data # The data value
self.next = None # Reference to the next node
Types of Linked Lists
- Singly Linked List: Each node points to the next node, and the last node points to
None
. - Doubly Linked List: Each node has two pointers—one to the next node and another to the previous node.
- Circular Linked List: The last node points back to the first node, forming a circular chain.
Diagram of Linked Lists:
Singly Linked List: [Data|Next] -> [Data|Next] -> [Data|None]
Doubly Linked List: None <- [Prev|Data|Next] <-> [Prev|Data|Next] -> None
Circular Linked List: [Data|Next] -> [Data|Next] -> [Data|Next] -> (back to first node)
Why Use Linked Lists?
- Dynamic Size: Unlike arrays, linked lists don’t require a fixed size during initialization. Nodes can be dynamically added or removed.
- Efficient Insertions/Deletions: Adding or removing elements in a linked list does not require shifting elements as in arrays.
- Flexibility: They allow for efficient memory usage as nodes are allocated as needed.
Common Operations on Linked Lists
1. Traversal
Visiting each node to access its data:
def traverse_linked_list(head):
current = head
while current is not None:
print(current.data, end=" -> ")
current = current.next
print("None")
# Example usage:
head = Node(1)
head.next = Node(2)
head.next.next = Node(3)
traverse_linked_list(head) # Output: 1 -> 2 -> 3 -> None
2. Insertion
Adding a node at the beginning, middle, or end:
def insert_at_end(head, data):
new_node = Node(data)
if head is None:
return new_node
current = head
while current.next is not None:
current = current.next
current.next = new_node
return head
# Example usage:
head = Node(1)
head = insert_at_end(head, 2)
head = insert_at_end(head, 3)
traverse_linked_list(head) # Output: 1 -> 2 -> 3 -> None
3. Deletion
Removing a node by value:
def delete_node(head, value):
if head is None:
return None
if head.data == value:
return head.next
current = head
while current.next is not None and current.next.data != value:
current = current.next
if current.next is not None:
current.next = current.next.next
return head
# Example usage:
head = Node(1)
head.next = Node(2)
head.next.next = Node(3)
head = delete_node(head, 2)
traverse_linked_list(head) # Output: 1 -> 3 -> None
4. Searching
Finding a node by its value:
def search_linked_list(head, target):
current = head
while current is not None:
if current.data == target:
return True
current = current.next
return False
# Example usage:
head = Node(1)
head.next = Node(2)
head.next.next = Node(3)
print(search_linked_list(head, 2)) # Output: True
print(search_linked_list(head, 4)) # Output: False
Advantages and Disadvantages
Advantages:
- Dynamic size.
- Efficient insertions and deletions.
- Better memory utilization for sparse datasets.
Disadvantages:
- No direct access to elements; must traverse the list.
- Overhead of pointers increases memory usage.
- More complex implementation compared to arrays.