A linked list is a linear collection of data elements, their order is not given by their physical placement in memory. Instead they are constructed in such a way that each element points to the next. It is a data structure that consists of a collection of nodes which together represents a sequence.
Nodes linked together
Linked lists can be singly or doubly linked.
Singly linked list
A singly linked list would use a node with data and only one reference (link) to the next node.
classNode:
def __init__(self, data):
self.data = data
self.next =None
Full implementation
classNode():
def __init__(self, data=None, next=None):
self.data = data
self.next = next
classLinkedList():
def __init__(self):
self.head =Nonedefappend(self, data):
ifnot self.head:
self.head = Node(data)
else:
next_node = self.head
while next_node.next:
next_node = next_node.next
next_node.next = Node(data)
defpush(self, data):
ifnot self.head:
self.head = Node(data)
else:
new_node = Node(data)
new_node.next = self.head
self.head = new_node
definsert(self, data, index):
ifnot self.head and index >0or index <0:
return"Index out of range."else:
next_node = self.head
prev_node =None curr_index =0while curr_index != index:
prev_node = next_node
next_node = next_node.next
curr_index +=1 new_node = Node(data)
if prev_node:
prev_node.next = new_node
new_node.next = next_node
else:
new_node.next = next_node
self.head = new_node
defpop(self):
next_node = self.head
prev_node =Nonewhile next_node.next:
prev_node = next_node
next_node = next_node.next
prev_node.next =Nonereturn next_node.data
defremove(self, index):
ifnot self.head and index >0or index <0:
return"Index out of range."else:
next_node = self.head
prev_node =None curr_index =0while curr_index != index:
prev_node = next_node
next_node = next_node.next
curr_index +=1if prev_node:
prev_node.next = next_node.next
else:
self.head = next_node.next
defreverse(self):
prev_node =None current = self.head
while current:
next_node = current.next
current.next = prev_node
prev_node = current
current = next_node
self.head = prev_node
defsearch(self, data):
ifnot self.head:
return"Not Found"else:
next_node = self.head
index =0while next_node.data != data:
next_node = next_node.next
index +=1ifnot next_node:
return"Not Found"return index
defsort(self):
current_node = self.head
next_node =Noneifnot self.head:
returnelse:
while current_node:
next_node = current_node.next
while next_node:
if current_node.data > next_node.data:
current_node.data, next_node.data = next_node.data, current_node.data
next_node = next_node.next
current_node = current_node.next
defprint_list(self):
next_node = self.head
while next_node:
print(next_node.data)
next_node = next_node.next
classNode {
constructor(data, next=null) {
this.data=data;
this.next=next;
}
}
classLinkedList {
constructor(head=null) {
this.head=head;
}
// add item to end of the list
append(item) {
if (this.head===null) {
this.head=newNode(item);
} else {
letnext=this.head;
while (next.next!==null) {
next=next.next;
}
next.next=newNode(item);
}
}
// add item to the front of the list
push(item) {
if (this.head===null) {
this.head=newNode(item);
} else {
lettemp=newNode(item);
temp.next=this.head;
this.head=temp;
}
}
// pop (remove) from end of the list
pop() {
letnext=this.head;
letprev=null;
while (next.next!==null) {
prev=next;
next=next.next;
}
prev.next=null;
returnnext.data;
}
// search if node value in list and return bool
searchNode(item) {
letnext=this.head;
while (next!==null) {
if (next.data===item) {
returntrue;
}
next=next.next;
}
returnfalse;
}
// traverse list and print each item
print_list() {
letnext=this.head;
while (next!==null) {
print(next.data);
next=next.next;
}
}
// turn list around
reverse() {
letnext=this.head;
letprev=null;
letcurr;
while (next!==null) {
curr=next;
next=curr.next;
curr.next=prev;
prev=curr;
}
this.head=prev;
}
}
usingnamespace std;
// Define Node class
classNode {
public:int data;
Node* next;
Node(int data) {
this->data = data;
this->next =nullptr;
}
};
// Define LinkedList class
classLinkedList {
private: Node* head;
public: LinkedList() {
head =nullptr;
}
// Append method
voidappend(int data) {
if (head ==nullptr) {
head =new Node(data);
} else {
Node* next_node = head;
while (next_node->next) {
next_node = next_node->next;
}
next_node->next =new Node(data);
}
}
// Push method
voidpush(int data) {
Node* new_node =new Node(data);
new_node->next = head;
head = new_node;
}
// Insert method
voidinsert(int data, int index) {
if (index <0) {
cout <<"Index out of range."<< endl;
return;
}
Node* next_node = head;
Node* prev_node =nullptr;
int curr_index =0;
while (next_node !=nullptr&& curr_index < index) {
prev_node = next_node;
next_node = next_node->next;
curr_index++;
}
if (next_node ==nullptr&& curr_index < index) {
cout <<"Index out of range."<< endl;
return;
}
Node* new_node =new Node(data);
if (prev_node) {
prev_node->next = new_node;
new_node->next = next_node;
} else {
new_node->next = next_node;
head = new_node;
}
}
// Pop method
intpop() {
if (head ==nullptr) {
cout <<"List is empty."<< endl;
return-1; // Return an error value
}
Node* next_node = head;
Node* prev_node =nullptr;
while (next_node->next) {
prev_node = next_node;
next_node = next_node->next;
}
int data = next_node->data;
if (prev_node) {
prev_node->next =nullptr;
} else {
head =nullptr;
}
delete next_node; // Free the memory of the popped node
return data;
}
// Remove method
voidremove(int index) {
if (index <0|| head ==nullptr) {
cout <<"Index out of range."<< endl;
return;
}
Node* next_node = head;
Node* prev_node =nullptr;
int curr_index =0;
while (next_node !=nullptr&& curr_index < index) {
prev_node = next_node;
next_node = next_node->next;
curr_index++;
}
if (next_node ==nullptr) {
cout <<"Index out of range."<< endl;
return;
}
if (prev_node) {
prev_node->next = next_node->next;
} else {
head = next_node->next;
}
delete next_node; // Free memory of removed node
}
// Reverse method
voidreverse() {
Node* prev_node =nullptr;
Node* current = head;
Node* next_node =nullptr;
while (current !=nullptr) {
next_node = current->next;
current->next = prev_node;
prev_node = current;
current = next_node;
}
head = prev_node;
}
// Search method
intsearch(int data) {
Node* next_node = head;
int index =0;
while (next_node !=nullptr) {
if (next_node->data == data) {
return index;
}
next_node = next_node->next;
index++;
}
return-1; // Return -1 if not found
}
// Sort method
voidsort() {
if (head ==nullptr) {
return;
}
Node* current_node = head;
Node* next_node =nullptr;
while (current_node !=nullptr) {
next_node = current_node->next;
while (next_node !=nullptr) {
if (current_node->data > next_node->data) {
swap(current_node->data, next_node->data);
}
next_node = next_node->next;
}
current_node = current_node->next;
}
}
// Print list method
voidprint_list() {
Node* next_node = head;
while (next_node !=nullptr) {
cout << next_node->data << endl;
next_node = next_node->next;
}
}
~LinkedList() {
Node* current = head;
Node* next_node =nullptr;
while (current !=nullptr) {
next_node = current->next;
delete current;
current = next_node;
}
}
};
classNode{int data; Node next;// Constructor
publicNode(int data){this.data= data;this.next=null;}}classLinkedList{ Node head;// Constructor
publicLinkedList(){ head =null;}// Append method
publicvoidappend(int data){if(head ==null){ head =new Node(data);}else{ Node nextNode = head;while(nextNode.next!=null){ nextNode = nextNode.next;} nextNode.next=new Node(data);}}// Push method
publicvoidpush(int data){ Node newNode =new Node(data); newNode.next= head; head = newNode;}// Insert method
publicvoidinsert(int data,int index){if(index <0){ System.out.println("Index out of range.");return;} Node nextNode = head; Node prevNode =null;int currIndex =0;while(nextNode !=null&& currIndex < index){ prevNode = nextNode; nextNode = nextNode.next; currIndex++;}if(nextNode ==null&& currIndex < index){ System.out.println("Index out of range.");return;} Node newNode =new Node(data);if(prevNode !=null){ prevNode.next= newNode; newNode.next= nextNode;}else{ newNode.next= nextNode; head = newNode;}}// Pop method
publicintpop(){if(head ==null){ System.out.println("List is empty.");return-1;// Error value
} Node nextNode = head; Node prevNode =null;while(nextNode.next!=null){ prevNode = nextNode; nextNode = nextNode.next;}int data = nextNode.data;if(prevNode !=null){ prevNode.next=null;}else{ head =null;}return data;}// Remove method
publicvoidremove(int index){if(index <0|| head ==null){ System.out.println("Index out of range.");return;} Node nextNode = head; Node prevNode =null;int currIndex =0;while(nextNode !=null&& currIndex < index){ prevNode = nextNode; nextNode = nextNode.next; currIndex++;}if(nextNode ==null){ System.out.println("Index out of range.");return;}if(prevNode !=null){ prevNode.next= nextNode.next;}else{ head = nextNode.next;}}// Reverse method
publicvoidreverse(){ Node prevNode =null; Node current = head; Node nextNode =null;while(current !=null){ nextNode = current.next; current.next= prevNode; prevNode = current; current = nextNode;} head = prevNode;}// Search method
publicintsearch(int data){ Node nextNode = head;int index =0;while(nextNode !=null){if(nextNode.data== data){return index;} nextNode = nextNode.next; index++;}return-1;// Not found
}// Sort method
publicvoidsort(){if(head ==null){return;} Node currentNode = head; Node nextNode =null;while(currentNode !=null){ nextNode = currentNode.next;while(nextNode !=null){if(currentNode.data> nextNode.data){int temp = currentNode.data; currentNode.data= nextNode.data; nextNode.data= temp;} nextNode = nextNode.next;} currentNode = currentNode.next;}}// Print list method
publicvoidprintList(){ Node nextNode = head;while(nextNode !=null){ System.out.println(nextNode.data); nextNode = nextNode.next;}}}
Doubly linked list
A doubly linked list would use a node with data and two references (links) which would point to the next and previous nodes.
Here’s a table summarizing the time complexities for various operations on single and double linked lists:
Operation
Single Linked List
Double Linked List
Access (by index)
O(n)
O(n)
Search (by value)
O(n)
O(n)
Insertion (beginning)
O(1)
O(1)
Insertion (end)
O(n)
O(1)
Insertion (middle)
O(n)
O(n)
Deletion (beginning)
O(1)
O(1)
Deletion (end)
O(n)
O(1)
Deletion (middle)
O(n)
O(n)
Traversal
O(n)
O(n)
Explanation:
Single Linked List: Each node only contains a reference to the next node, so operations that involve traversing the list or accessing an arbitrary index generally take O(n) time.
Double Linked List: Each node contains references to both the next and previous nodes. This makes certain operations, like deletion from the end or inserting at the end, more efficient (O(1)) compared to a singly linked list. However, accessing an arbitrary index still requires traversal, so it remains O(n) for both types of lists.