Given the head of a singly linked list containing integers, reverse the nodes of the list in groups of k and return the head of the modified list. If the number of nodes is not a multiple of k, then the remaining nodes at the end should be kept as is and not reversed.
Do not change the values of the nodes, only change the links between nodes.
Input: head -> 1 -> 2 -> 3 -> 4 -> 5, k = 2
Output: head -> 2 -> 1 -> 4 -> 3 -> 5
Explanation: The groups 1 -> 2 and 3 -> 4 were reversed as 2 -> 1 and 4 -> 3.
Input: head -> 1 -> 2 -> 3 -> 4 -> 5, k = 3
Output: head -> 3 -> 2 -> 1 -> 4 -> 5
Explanation: The groups 1 -> 2 -> 3 were reversed as 3 -> 2 -> 1.
Note that 4 -> 5 was not reversed.
Input: head -> 6 -> 1 -> 2 -> 3 -> 4 -> 7, k = 4
The intuition is to reverse the nodes of the linked list in groups of k
by identifying each group, reversing the links within that group, and then connecting the reversed groups, leaving any remaining nodes as they are if the group is less than k
.
First, traverse the linked list to identify segments of K nodes. For each segment, adjust the pointers within the segment to reverse the direction of the nodes. This involves iterating through the segment and changing the links between nodes.
Next, after reversing a segment, connect the reversed segment to the previous part of the list. Continue this process until you reach the end of the list.
Finally, if there are fewer than K nodes left at the end of the list, leave them as they are. Return the head of the modified linked list.
temp
to the head of the linked list. Using temp
, traverse to the Kth node iteratively.nextNode
and set the Kth node’s next pointer to null
. This effectively breaks the linked list into a smaller list of size K that can be reversed and attached back.temp
to the Kth node as an individual linked list and reverse it. This can be done using a helper function that reverses the linked list.temp
now at its tail and the Kth node pointing to its head. Update temp
's next
pointer to nextNode
. If reversing the first segment of K nodes, update the head to the Kth node.prevLast
pointer to maintain the link between the end of the previous reversed segment and the current segment.#include <bits/stdc++.h>
using namespace std;
// Definition of singly linked list:
struct ListNode
{
int val;
ListNode *next;
ListNode()
{
val = 0;
next = NULL;
}
ListNode(int data1)
{
val = data1;
next = NULL;
}
ListNode(int data1, ListNode *next1)
{
val = data1;
next = next1;
}
};
class Solution {
public:
// Function to reverse a linked list
// Using the 3-pointer approach
ListNode* reverseLinkedList(ListNode *head)
{
/* Initialize 'temp' at
* head of linked list */
ListNode* temp = head;
/* Initialize pointer 'prev'
* to NULL, representing
* the previous node */
ListNode* prev = NULL;
// Continue till 'temp'
// reaches the end (NULL)
while(temp != NULL){
/* Store the next node in 'front'
* to preserve the reference */
ListNode* front = temp->next;
/* Reverse the direction of the
* current node's 'next' pointer
* to point to 'prev' */
temp->next = prev;
/* Move 'prev' to the current
* node for the next iteration */
prev = temp;
/* Move 'temp' to the 'front' node
* advancing the traversal */
temp = front;
}
// Return the new head
// of the reversed linked list
return prev;
}
// Function to get the Kth node from a
// given position in the linked list
ListNode* getKthNode(ListNode* temp, int k){
// Decrement K
// as we already start
// from the 1st node
k -= 1;
// Decrement K until it reaches the desired position
while(temp != NULL && k > 0){
// Decrement k as temp progresses
k--;
// Move to the next node
temp = temp -> next;
}
// Return the Kth node
return temp;
}
// Function to reverse nodes in groups of K
ListNode* reverseKGroup(ListNode* head, int k){
/* Initialize a temporary
* node to traverse the list */
ListNode* temp = head;
/* Initialize a pointer to track
* the last node of the previous group */
ListNode* prevLast = NULL;
// Traverse through the linked list
while(temp != NULL){
// Get the Kth node of the current group
ListNode* kThNode = getKthNode(temp, k);
/* If the Kth node is NULL
* (not a complete group) */
if(kThNode == NULL){
/* If there was a previous group,
* link the last node to the current node */
if(prevLast){
prevLast -> next = temp;
}
// Exit the loop
break;
}
/* Store the next node
* after the Kth node */
ListNode* nextNode = kThNode -> next;
/* Disconnect the Kth node
* to prepare for reversal */
kThNode -> next = NULL;
// Reverse the nodes from temp to the Kth node
reverseLinkedList(temp);
/* Adjust the head if the reversal
* starts from the head */
if(temp == head){
head = kThNode;
}else{
/* Link the last node of the previous
* group to the reversed group */
prevLast -> next = kThNode;
}
/* Update the pointer to the
* last node of the previous group */
prevLast = temp;
// Move to the next group
temp = nextNode;
}
// Return the head of the modified linked list
return head;
}
};
// Function to print the linked list
void printLinkedList(ListNode* head) {
ListNode* temp = head;
while (temp != nullptr) {
cout << temp->val << " ";
temp = temp->next;
}
cout << endl;
}
int main() {
// Create a linked list with values 5, 4, 3, 7, 9 and 2
ListNode* head = new ListNode(5);
head->next = new ListNode(4);
head->next->next = new ListNode(3);
head->next->next->next = new ListNode(7);
head->next->next->next->next = new ListNode(9);
head->next->next->next->next->next = new ListNode(2);
// Print the original linked list
cout << "Original Linked List: ";
printLinkedList(head);
// Reverse the linked list in groups of K
Solution solution;
head = solution.reverseKGroup(head, 4);
// Print the reversed linked list
cout << "Reversed Linked List: ";
printLinkedList(head);
return 0;
}
import java.util.*;
// Definition of singly linked list:
class ListNode {
int val;
ListNode next;
ListNode() {
val = 0;
next = null;
}
ListNode(int data1) {
val = data1;
next = null;
}
ListNode(int data1, ListNode next1) {
val = data1;
next = next1;
}
}
class Solution {
// Function to reverse a linked list
// using the 3-pointer approach
public ListNode reverseLinkedList(ListNode head) {
/* Initialize 'temp' at
* head of linked list */
ListNode temp = head;
/* Initialize pointer 'prev'
* to NULL, representing
* the previous node */
ListNode prev = null;
// Continue till 'temp'
// reaches the end (NULL)
while (temp != null) {
/* Store the next node in 'front'
* to preserve the reference */
ListNode front = temp.next;
/* Reverse the direction of the
* current node's 'next' pointer
* to point to 'prev' */
temp.next = prev;
/* Move 'prev' to the current
* node for the next iteration */
prev = temp;
/* Move 'temp' to the 'front' node
* advancing the traversal */
temp = front;
}
// Return the new head
// of the reversed linked list
return prev;
}
// Function to get the Kth node from a
// given position in the linked list
public ListNode getKthNode(ListNode temp, int k) {
// Decrement K
// as we already start
// from the 1st node
k -= 1;
// Decrement K until it reaches the desired position
while (temp != null && k > 0) {
// Decrement k as temp progresses
k--;
// Move to the next node
temp = temp.next;
}
// Return the Kth node
return temp;
}
// Function to reverse nodes in groups of K
public ListNode reverseKGroup(ListNode head, int k) {
/* Initialize a temporary
* node to traverse the list */
ListNode temp = head;
/* Initialize a pointer to track
* the last node of the previous group */
ListNode prevLast = null;
// Traverse through the linked list
while (temp != null) {
// Get the Kth node of the current group
ListNode kThNode = getKthNode(temp, k);
/* If the Kth node is NULL
* (not a complete group) */
if (kThNode == null) {
/* If there was a previous group,
* link the last node to the current node */
if (prevLast != null) {
prevLast.next = temp;
}
// Exit the loop
break;
}
/* Store the next node
* after the Kth node */
ListNode nextNode = kThNode.next;
/* Disconnect the Kth node
* to prepare for reversal */
kThNode.next = null;
// Reverse the nodes from temp to the Kth node
reverseLinkedList(temp);
/* Adjust the head if the reversal
* starts from the head */
if (temp == head) {
head = kThNode;
} else {
/* Link the last node of the previous
* group to the reversed group */
prevLast.next = kThNode;
}
/* Update the pointer to the
* last node of the previous group */
prevLast = temp;
// Move to the next group
temp = nextNode;
}
// Return the head of the modified linked list
return head;
}
// Function to print the linked list
public static void printLinkedList(ListNode head) {
ListNode temp = head;
while (temp != null) {
System.out.print(temp.val + " ");
temp = temp.next;
}
System.out.println();
}
public static void main(String[] args) {
// Create a linked list with values 5, 4, 3, 7, 9 and 2
ListNode head = new ListNode(5);
head.next = new ListNode(4);
head.next.next = new ListNode(3);
head.next.next.next = new ListNode(7);
head.next.next.next.next = new ListNode(9);
head.next.next.next.next.next = new ListNode(2);
// Print the original linked list
System.out.print("Original Linked List: ");
printLinkedList(head);
// Reverse the linked list in groups of K
Solution solution = new Solution();
head = solution.reverseKGroup(head, 4);
// Print the reversed linked list
System.out.print("Reversed Linked List: ");
printLinkedList(head);
}
}
# Definition of singly linked list:
class ListNode:
def __init__(self, val=0, next=None):
self.val = val
self.next = next
class Solution:
# Function to reverse a linked list
# using the 3-pointer approach
def reverseLinkedList(self, head):
""" Initialize 'temp' at
head of linked list """
temp = head
""" Initialize pointer 'prev'
to NULL, representing
the previous node """
prev = None
# Continue till 'temp'
# reaches the end (NULL)
while temp:
""" Store the next node in 'front'
to preserve the reference """
front = temp.next
""" Reverse the direction of the
current node's 'next' pointer
to point to 'prev' """
temp.next = prev
""" Move 'prev' to the current
node for the next iteration """
prev = temp
""" Move 'temp' to the 'front' node
advancing the traversal """
temp = front
# Return the new head
# of the reversed linked list
return prev
# Function to get the Kth node from a
# given position in the linked list
def getKthNode(self, temp, k):
# Decrement K
# as we already start
# from the 1st node
k -= 1
# Decrement K until it reaches the desired position
while temp and k > 0:
# Decrement k as temp progresses
k -= 1
# Move to the next node
temp = temp.next
# Return the Kth node
return temp
# Function to reverse nodes in groups of K
def reverseKGroup(self, head, k):
""" Initialize a temporary
node to traverse the list """
temp = head
""" Initialize a pointer to track
the last node of the previous group """
prevLast = None
# Traverse through the linked list
while temp:
# Get the Kth node of the current group
kThNode = self.getKthNode(temp, k)
""" If the Kth node is NULL
(not a complete group) """
if not kThNode:
""" If there was a previous group,
link the last node to the current node """
if prevLast:
prevLast.next = temp
# Exit the loop
break
""" Store the next node
after the Kth node """
nextNode = kThNode.next
""" Disconnect the Kth node
to prepare for reversal """
kThNode.next = None
# Reverse the nodes from temp to the Kth node
self.reverseLinkedList(temp)
""" Adjust the head if the reversal
starts from the head """
if temp == head:
head = kThNode
else:
""" Link the last node of the previous
group to the reversed group """
prevLast.next = kThNode
""" Update the pointer to the
last node of the previous group """
prevLast = temp
# Move to the next group
temp = nextNode
# Return the head of the modified linked list
return head
# Function to print the linked list
def printLinkedList(head):
temp = head
while temp:
print(temp.val, end=" ")
temp = temp.next
print()
# Create a linked list with values 5, 4, 3, 7, 9 and 2
head = ListNode(5)
head.next = ListNode(4)
head.next.next = ListNode(
// Definition of singly linked list:
class ListNode {
constructor(val = 0, next = null) {
this.val = val;
this.next = next;
}
}
class Solution {
// Function to reverse a linked list
// using the 3-pointer approach
reverseLinkedList(head) {
/* Initialize 'temp' at
* head of linked list */
let temp = head;
/* Initialize pointer 'prev'
* to NULL, representing
* the previous node */
let prev = null;
// Continue till 'temp'
// reaches the end (NULL)
while (temp !== null) {
/* Store the next node in 'front'
* to preserve the reference */
let front = temp.next;
/* Reverse the direction of the
* current node's 'next' pointer
* to point to 'prev' */
temp.next = prev;
/* Move 'prev' to the current
* node for the next iteration */
prev = temp;
/* Move 'temp' to the 'front' node
* advancing the traversal */
temp = front;
}
// Return the new head
// of the reversed linked list
return prev;
}
// Function to get the Kth node from a
// given position in the linked list
getKthNode(temp, k) {
// Decrement K
// as we already start
// from the 1st node
k -= 1;
// Decrement K until it reaches the desired position
while (temp !== null && k > 0) {
// Decrement k as temp progresses
k--;
// Move to the next node
temp = temp.next;
}
// Return the Kth node
return temp;
}
// Function to reverse nodes in groups of K
reverseKGroup(head, k) {
/* Initialize a temporary
* node to traverse the list */
let temp = head;
/* Initialize a pointer to track
* the last node of the previous group */
let prevLast = null;
// Traverse through the linked list
while (temp !== null) {
// Get the Kth node of the current group
let kThNode = this.getKthNode(temp, k);
/* If the Kth node is NULL
* (not a complete group) */
if (kThNode === null) {
/* If there was a previous group,
* link the last node to the current node */
if (prevLast !== null) {
prevLast.next = temp;
}
// Exit the loop
break;
}
/* Store the next node
* after the Kth node */
let nextNode = kThNode.next;
/* Disconnect the Kth node
* to prepare for reversal */
kThNode.next = null;
// Reverse the nodes from temp to the Kth node
this.reverseLinkedList(temp);
/* Adjust the head if the reversal
* starts from the head */
if (temp === head) {
head = kThNode;
} else {
/* Link the last node of the previous
* group to the reversed group */
prevLast.next = kThNode;
}
/* Update the pointer to the
* last node of the previous group */
prevLast = temp;
// Move to the next group
temp = nextNode;
}
// Return the head of the modified linked list
return head;
}
}
// Function to print the linked list
function printLinkedList(head) {
let temp = head;
while (temp !== null) {
process.stdout.write(temp.val + " ");
temp = temp.next;
}
console.log();
}
// Create a linked list with values 5, 4, 3, 7, 9 and 2
let head = new ListNode(5);
head.next = new ListNode(4);
head.next.next = new ListNode(3);
head.next.next.next = new ListNode(7);
head.next.next.next.next = new ListNode(9);
head.next.next.next.next.next = new ListNode(2);
// Print the original linked list
process.stdout.write("Original Linked List: ");
printLinkedList(head);
// Reverse the linked list in groups of K
const solution = new Solution();
head = solution.reverseKGroup(head, 4);
// Print the reversed linked list
process.stdout.write("Reversed Linked List: ");
printLinkedList(head);
Time Complexity: O(2N) because it consists of actions of reversing segments of K and finding the Kth node, both of which operate in linear time. Thus, O(N) + O(N) = O(2N), which simplifies to O(N).
Space Complexity: O(1) because the code operates in place without any additional space requirements.
Q: What happens if n is not a multiple of k?
A: The last remaining nodes (less than k) are not reversed.
Q: What if we needed to reverse only alternate k-groups?
A: Introduce a flip flag: Reverse when flip = true Skip when flip = false.
Q: How does this compare to recursive reversal?
A: Recursive method (O(n) space) adds function calls. Iterative (O(1) space) is preferred for large inputs.