-
Notifications
You must be signed in to change notification settings - Fork 243
Swap Nodes In Pairs
Raymond Chen edited this page Jun 15, 2023
·
7 revisions
- 🔗 Leetcode Link: Swap Nodes in Pairs
- 💡 Problem Difficulty: Medium
- ⏰ Time to complete: 10 to 12 mins
- 🛠️ Topics: Linked Lists, Recursion
- 🗒️ Similar Questions: Swapping Nodes in a Linked List
Understand what the interviewer is asking for by using test cases and questions about the problem.
- Established a set (2-3) of test cases to verify their own solution later.
- Established a set (1-2) of edge cases to verify their solution handles complexities.
- Have fully understood the problem and have no clarifying questions.
- Have you verified any Time/Space Constraints for this problem?
Be sure that you clarify the input and output parameters of the problem:
- Could the input node be null?
- I think we should account for that. It’d be a standard edge case in an interview setting.
- Is the list guaranteed to have an even length?
- The list can be any length, even just 1 node.
- The numerical value in the nodes does not matter, right?
- The solution to this problem shouldn’t have to worry about specific node values.
Run through a set of example cases:
HAPPY CASE
Input: head = [1,2,3,4]
Output: [2,1,4,3]
Input: head = []
Output: []
EDGE CASE
Input: head = [1]
Output: [1]
Match what this problem looks like to known categories of problems, e.g. Linked List or Dynamic Programming, and strategies or patterns in those categories.
- Multiple passes: To find the length, or save other information about the contents. If we were able to take multiple passes of the linked list, would that help solve the problem? Multiple passes would not help this problem since we aren’t collecting unique items or need any pre-processing.
- Two pointers: ‘Race car’ strategy with one regular pointer, and one fast pointer. If we used two pointers to iterate through the list at different speeds, would that help us solve this problem? A slow and fast pointer may not help the problem.
- Temp node: Helpful for preventing errors when returning ‘head’ if merging lists, deleting from lists. Would using a temp head as a starting point help simplify our code and handle edge cases? Since we have to manipulate the order of the list, including the head of the list, a temp head would help maintain a pointer to beginning of the list.
Plan the solution with appropriate visualizations and pseudocode.
1) Establish a temp head that points to the first node
2) Keep a pointer to the temp head
3) Iterate through the list while the pointer has TWO valid nodes ahead
4) Swap the two nodes ahead of the pointer
5) Move the pointer to the second post-swap node
6) Continue to iterate until end of list
- The trick is updating the pointers carefully. Key takeaway is: to avoid mistakes first update the deepest pointers, e.g. .next.next, then more shallow ones, etc. Try using a simple recursive technique to swap adjacent pairs. We swap the 2 adjacent nodes and find the next of swapped second adjacent node through recursion. Base case occurs when either the node passed contains null value aur adjacent node of passed node does not exist.
Implement the code to solve the algorithm.
class Solution:
def swapPairs(self, head: Optional[ListNode]) -> Optional[ListNode]:
# initialize one temp node and mark it as prev node
prev = tempHead = ListNode(0, next = head)
# check the next node and next next node exist
while prev.next and prev.next.next:
# get the next node and next next node exist
first = prev.next
second = prev.next.next
# set first.next to second.next, point to next node pair
first.next = second.next
# set second.next to first, perform the swap
second.next = first
# set prev.next to second, point to current node pair
prev.next = second
# set prev to first, set prev to node previous to next node pair
prev = first
return tempHead.next
class Solution {
public ListNode swapPairs(ListNode head) {
ListNode d = new ListNode(-1);
d.next = head;
// initialize one temp node and mark it as prev node
ListNode prev = d;
while(head != null && head.next!=null){
// get the current node and next node
ListNode a = head;
ListNode b = head.next;
// now previous node's next node becomes b
prev.next = b;
// connect b nodes next to a nodes next to preserve chaining
a.next = b.next;
// to make a node comes after b,
b.next = a;
// now a node becomes previous
prev = a;
// iterate to next pair of nodes
head= a.next;
}
return d.next;
}
}
Review the code by running specific example(s) and recording values (watchlist) of your code's variables along the way.
- Trace through your code with an input to check for the expected output
- Catch possible edge cases and off-by-one errors
Evaluate the performance of your algorithm and state any strong/weak or future potential work.
- Time Complexity: O(N), as we used a single pass-through linked list
- Space Complexity: O(1), as only 1 temp node created for any size list