92. Reverse Linked List II
Problem Description
The problem presents us with a singly linked list and two integers, left
and right
, with the condition that left <= right
. The goal is to reverse the nodes in the linked list that fall between the positions left
and right
(inclusive). The positions are 1-indexed, not 0-indexed, so the first node in the list would be position 1, the second node would be position 2, and so on. Ultimately, the modified linked list should be returned with the specified portion reversed while keeping the rest of the list's original structure intact.
Intuition
To tackle this problem, we need to understand the concept of reversing a linked list and also keeping track of the nodes at the boundaries of the section we want to reverse. The core idea is to iterate through the linked list to locate the node just before the left
position (the start of our reversal) and the node at the right
position (the end of our reversal).
Upon reaching the left
position, we will begin the process of reversing the links between the nodes until we reach the right
position. The key points are:
- We need a reference to the node just before the
left
position to reconnect the reversed sublist back to the preceding part of the list. - We need to store the node at the
left
position as it will become the tail of the reversed sublist and connect to the node following theright
position.
This can be achieved with a few pointers and careful reassignments of the next
pointers within the sublist. By keeping track of the current node being processed and the previous node within the reversal range, we can reverse the links one by one. Finally, we must ensure we reattach the reversed sublist to the non-reversed parts properly to maintain a functioning linked list.
Learn more about Linked List patterns.
Solution Approach
The solution employs two essential steps: iterating to the specified nodes and reversing the sublist. Here, we use a dummy node to simplify edge case handling, such as when reversing from the first node.
-
Initialization
- A dummy node is created with its
next
pointing to the head of the list. This helps manage the edge case where theleft
is 1, indicating the reversal starting from the head. - Two pointers,
pre
andcur
, are initially set to the dummy and head nodes, respectively.
- A dummy node is created with its
-
Locating the Start Point
- We move the
pre
pointerleft - 1
times forward to reach the node just before where the reversal is supposed to start. - At this point,
pre.next
points to the first node to be reversed. - We also set a pointer
q
to mark the beginning of the sublist to be reversed (pre.next
).
- We move the
-
Reversal Process
- A loop runs
right - left + 1
times, which corresponds to the length of the sublist to be reversed. - Within the loop, we perform the reversal. We constantly update the
cur.next
pointer to point topre
, effectively reversing the link between the current pair of nodes. - After reversing the link, we need to update the
pre
andcur
pointers.pre
moves to wherecur
used to be, andcur
shifts to the next node in the original sequence using the temporary pointert
which holds the unreversed remainder of the list.
- A loop runs
-
Final Connections
- After the loop, the sublist is reversed. However, we still need to connect the reversed sublist back into the main list.
- The pointer
p.next
is set topre
, which, after the loop termination, points to theright
node that is now the head of the reversed sublist. - The initial
left
node, which is now at the end of the reversed sublist and pointed to byq
, should point to thecur
node, which is the node right after theright
position orNone
ifright
was at the end of the list.
-
Return the Result
- The function returns
dummy.next
as the new head of the list, ensuring that whether we reversed from the head or any other part of the list, we have the right starting point.
- The function returns
In summary, the solution takes a careful approach to change pointers and reconnect the nodes to achieve the desired reversal between the left
and right
indices, while keeping the rest of the list intact.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's illustrate the solution approach with a small example. Assume we have a linked list with elements 1 → 2 → 3 → 4 → 5, and we are asked to reverse the nodes between left = 2
and right = 4
. The positions of these elements in the list are: 1 → 2 → 3 → 4 → 5.
Following the solution approach step by step:
-
Initialization
- Create a dummy node (
pre
) which points to the head of the list. - Initialize another pointer (
cur
) to the head which is the first node (1
).
- Create a dummy node (
-
Locating the Start Point
- Since
left = 2
, we move thepre
pointerleft - 1
(1 time) forward, and it now points to node1
. - The
pre.next
then points to node2
, which is the start of the sublist we want to reverse. - We set a pointer
q
to mark the beginning of the sublist to be reversed (pre.next
), soq
points to node2
.
- Since
-
Reversal Process
- We need to reverse the sublist from position 2 to 4. The loop will run
right - left + 1
(4 - 2 + 1 = 3
times). - In the first loop iteration,
cur
points to2
and itsnext
is3
. We temporarily store the node followingcur
in pointert
(node3
), then setcur.next
topre
(node1
), and updatepre
tocur
(node2
). Nowcur
points tot
(node3
). - We repeat this process for node
3
and4
. Upon completion of the loop, our list looks like 1 → 2 ← 3 ← 4 withpre
at4
andcur
pointing to5
.
- We need to reverse the sublist from position 2 to 4. The loop will run
-
Final Connections
- We need to make the final connections to integrate the reversed sublist with the rest of the list.
- Connect
p.next
(the original start node1
) topre
(which is now4
), andq.next
(which holds the start of the reversed sublist2
) tocur
(node5
which is the remaining part of the list)
-
Return the Result
- Return the
dummy.next
, which points to the new head of the list (node1
).
- Return the
The modified linked list will now look like 1 → 4 → 3 → 2 → 5, with the nodes between position 2
and 4
reversed.
Solution Implementation
1class ListNode:
2 def __init__(self, val=0, next=None):
3 self.val = val
4 self.next = next
5
6class Solution:
7 def reverseBetween(self, head: Optional[ListNode], left: int, right: int) -> Optional[ListNode]:
8 # If the list only contains one node or no reversal is needed, return the head as is.
9 if head.next is None or left == right:
10 return head
11
12 # Initialize a dummy node to simplify edge cases
13 # where the reversal might include the head of the list.
14 dummy = ListNode(0)
15 dummy.next = head
16
17 # This node will eventually point to the node right before
18 # the reversal starts. Initialize it to the dummy node.
19 predecessor = dummy
20
21 # Move the predecessor to the node right before where
22 # the reversal is supposed to start.
23 for _ in range(left - 1):
24 predecessor = predecessor.next
25
26 # Initialize the 'reverse_start' node, which will eventually point to the
27 # first node in the sequence that needs to be reversed.
28 reverse_start = predecessor.next
29
30 # The 'current' node will traverse the sublist that needs to be reversed.
31 current = reverse_start
32
33 # This loop reverses the nodes between 'left' and 'right'.
34 # 'next_temp' is used to temporarily store the next node as we
35 # rearrange pointers.
36 for _ in range(right - left + 1):
37 next_temp = current.next
38 current.next = predecessor
39 predecessor, current = current, next_temp
40
41 # Link the nodes preceding the reversed sublist to the first node
42 # in the reversed sequence.
43 predecessor.next = predecessor
44
45 # Link the last node in the reversed sublist to the remaining
46 # part of the list that was not reversed.
47 reverse_start.next = current
48
49 # Return the new head of the list, which is the next of dummy node.
50 return dummy.next
51
1/**
2 * Definition for singly-linked list.
3 */
4class ListNode {
5 int val;
6 ListNode next;
7 ListNode() {}
8 ListNode(int val) { this.val = val; }
9 ListNode(int val, ListNode next) { this.val = val; this.next = next; }
10}
11
12class Solution {
13
14 /**
15 * Reverses a section of a singly-linked list between the given positions.
16 *
17 * @param head The head of the linked list.
18 * @param left The position from where to start the reversal (1-indexed).
19 * @param right The position where to end the reversal (1-indexed).
20 * @return The head of the modified linked list.
21 */
22 public ListNode reverseBetween(ListNode head, int left, int right) {
23 // If there is only one node or no need to reverse, return the original list.
24 if (head.next == null || left == right) {
25 return head;
26 }
27
28 // Dummy node to simplify the handling of the head node.
29 ListNode dummyNode = new ListNode(0, head);
30
31 // Pointer to track the node before the reversal section.
32 ListNode nodeBeforeReverse = dummyNode;
33 for (int i = 0; i < left - 1; ++i) {
34 nodeBeforeReverse = nodeBeforeReverse.next;
35 }
36
37 // 'firstReversed' will become the last node after the reversal.
38 ListNode firstReversed = nodeBeforeReverse.next;
39 // 'current' is used to track the current node being processed.
40 ListNode current = firstReversed;
41 ListNode prev = null;
42
43 // Perform the actual reversal between 'left' and 'right'.
44 for (int i = 0; i < right - left + 1; ++i) {
45 ListNode nextTemp = current.next;
46 current.next = prev;
47 prev = current;
48 current = nextTemp;
49 }
50
51 // Reconnect the reversed section back to the list.
52 nodeBeforeReverse.next = prev; // Connect with node before reversed part.
53 firstReversed.next = current; // Connect the last reversed node to the remainder of the list.
54
55 // Return the new head of the list.
56 return dummyNode.next;
57 }
58}
59
1/**
2 * Definition for singly-linked list.
3 * struct ListNode {
4 * int val;
5 * ListNode *next;
6 * ListNode() : val(0), next(nullptr) {}
7 * ListNode(int x) : val(x), next(nullptr) {}
8 * ListNode(int x, ListNode *next) : val(x), next(next) {}
9 * };
10 */
11class Solution {
12public:
13 ListNode* reverseBetween(ListNode* head, int left, int right) {
14 // If there is only one node or no node to reverse
15 if (!head || left == right) {
16 return head;
17 }
18
19 // Create a dummy node to handle edge cases, such as reversing the head node
20 ListNode* dummyNode = new ListNode(0);
21 dummyNode->next = head;
22
23 // Pointers for the node before the reversing part and the first node to reverse
24 ListNode* preReverse = dummyNode;
25
26 // Iterate to find the node before the left position
27 for (int i = 0; i < left - 1; ++i) {
28 preReverse = preReverse->next;
29 }
30
31 // Start reversing from the left position
32 ListNode* current = preReverse->next;
33 ListNode* nextNode = nullptr;
34 ListNode* prev = nullptr;
35
36 // Apply the reverse from left to right positions
37 for (int i = 0; i < right - left + 1; ++i) {
38 nextNode = current->next; // Save the next node to move on
39 current->next = prev; // Reverse the link
40 prev = current; // Move prev one step forward for the next iteration
41 current = nextNode; // Move to the next node in the list
42 }
43
44 // Adjust the links for the node before left and the node right after the reversed part
45 preReverse->next->next = current; // Connect the reversed part with the rest of the list
46 preReverse->next = prev; // Connect the start of the reversed list to the previous part
47
48 // Return the new head of the list
49 ListNode* newHead = dummyNode->next;
50 delete dummyNode; // Clean up the memory used by dummyNode
51 return newHead;
52 }
53};
54
1// Definition for singly-linked list.
2class ListNode {
3 val: number;
4 next: ListNode | null;
5 constructor(val: number = 0, next: ListNode | null = null) {
6 this.val = val;
7 this.next = next;
8 }
9}
10
11/**
12 * Reverses a portion of the singly-linked list between positions 'left' and 'right'.
13 *
14 * @param {ListNode | null} head The head of the linked list.
15 * @param {number} left The position to start reversing from (1-indexed).
16 * @param {number} right The position to stop reversing at (1-indexed).
17 * @return {ListNode | null} The head of the modified list.
18 */
19function reverseBetween(head: ListNode | null, left: number, right: number): ListNode | null {
20 // Base case: If the sublist to reverse is of size 0, return the original list.
21 const sublistLength = right - left;
22 if (sublistLength === 0) {
23 return head;
24 }
25
26 // Create a dummy node to handle edge cases seamlessly.
27 const dummyNode = new ListNode(0, head);
28 let previousNode: ListNode | null = null;
29 let currentNode: ListNode | null = dummyNode;
30
31 // Move the currentNode to the position right before where reversal begins.
32 for (let i = 0; i < left; i++) {
33 previousNode = currentNode;
34 currentNode = currentNode.next;
35 }
36
37 // The previousNode now points to the node right before the start of the sublist.
38 const sublistHeadPrev = previousNode;
39 previousNode = null;
40
41 // Reverse the sublist from the 'left' to 'right' position.
42 for (let i = 0; i <= sublistLength; i++) {
43 const nextNode = currentNode.next;
44 currentNode.next = previousNode;
45 previousNode = currentNode;
46 currentNode = nextNode;
47 }
48
49 // Connect the reversed sublist back to the unchanged part of the original list.
50 sublistHeadPrev.next.next = currentNode;
51 sublistHeadPrev.next = previousNode;
52
53 // Return the dummy node's next, which is the new head of the linked list.
54 return dummyNode.next;
55}
56
Time and Space Complexity
The time complexity of the given code can be determined by analyzing the number of individual operations that are performed as the input size (the size of the linked list) grows. The reversal operation within the section of the linked list bounded by left
and right
is the most significant part of the function.
- The code iterates from the dummy node to the node just before where the reversal starts (
left - 1
iterations), which isO(left)
. - Then it reverses the nodes between the
left
andright
position, takingright - left + 1
iterations, which isO(right - left)
.
Assuming n
is the total number of nodes in the linked list, the time complexity is the sum of the two:
O(left) + O(right - left)
, which is equivalent to O(right)
. Since right
is at most n
, the upper bound for the time complexity is O(n)
.
For space complexity, the code only uses a fixed number of extra variables (dummy
, pre
, p
, q
, cur
, and t
), irrespective of the input size. These variables hold references to nodes in the list but do not themselves result in additional space that scales with the input size. Therefore, the space complexity is O(1)
, meaning it is constant.
Learn more about how to find time and space complexity quickly using problem constraints.
What does the following code do?
1def f(arr1, arr2):
2 i, j = 0, 0
3 new_arr = []
4 while i < len(arr1) and j < len(arr2):
5 if arr1[i] < arr2[j]:
6 new_arr.append(arr1[i])
7 i += 1
8 else:
9 new_arr.append(arr2[j])
10 j += 1
11 new_arr.extend(arr1[i:])
12 new_arr.extend(arr2[j:])
13 return new_arr
14
1public static List<Integer> f(int[] arr1, int[] arr2) {
2 int i = 0, j = 0;
3 List<Integer> newArr = new ArrayList<>();
4
5 while (i < arr1.length && j < arr2.length) {
6 if (arr1[i] < arr2[j]) {
7 newArr.add(arr1[i]);
8 i++;
9 } else {
10 newArr.add(arr2[j]);
11 j++;
12 }
13 }
14
15 while (i < arr1.length) {
16 newArr.add(arr1[i]);
17 i++;
18 }
19
20 while (j < arr2.length) {
21 newArr.add(arr2[j]);
22 j++;
23 }
24
25 return newArr;
26}
27
1function f(arr1, arr2) {
2 let i = 0, j = 0;
3 let newArr = [];
4
5 while (i < arr1.length && j < arr2.length) {
6 if (arr1[i] < arr2[j]) {
7 newArr.push(arr1[i]);
8 i++;
9 } else {
10 newArr.push(arr2[j]);
11 j++;
12 }
13 }
14
15 while (i < arr1.length) {
16 newArr.push(arr1[i]);
17 i++;
18 }
19
20 while (j < arr2.length) {
21 newArr.push(arr2[j]);
22 j++;
23 }
24
25 return newArr;
26}
27
Recommended Readings
Linked List Cycle Given a linked list with potentially a loop determine whether the linked list from the first node contains a cycle in it For bonus points do this with constant space Parameters nodes The first node of a linked list with potentially a loop Result Whether there is a loop contained
Patterns The Shortest Path Algorithm for Coding Interviews The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we can determine the
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Got a question? Ask the Monster Assistant anything you don't understand.
Still not clear? Submit the part you don't understand to our editors. Or join our Discord and ask the community.