1047. Remove All Adjacent Duplicates In String
Problem Description
In this problem, you're presented with a string s
that contains only lowercase English letters. Your task is to repetitively remove pairs of adjacent, identical characters until no such pairs remain. The process is to look for two adjacent letters that are the same, and then remove them from the string. This is done over and over until it's no longer possible to do so. The result is the final, reduced string with no adjacent pairs of the same character. The uniqueness of the answer is guaranteed, which means no matter how you remove the pairs, the final string will be the same.
Intuition
The idea behind solving this problem involves simulating the described process using a data structure that allows for efficient addition and removal of elements, mainly when these operations are performed at one end. A stack is perfect for this case because it allows us to add elements to the top and also remove from the top, which can be thought of as the end of the string we are building.
As you iterate through the string, you examine each character. If the current character is the same as the one at the top of the stack (meaning they are adjacent duplicates in the context of the string), you remove the character from the stack (simulating the removal of the adjacent duplicate). If the current character is not a duplicate, or if the stack is empty, you push the current character onto the stack.
The process is akin to folding paper; whenever you spot a foldable (matchable) pair, you fold (remove) it. By the end of the string, the stack will contain all the characters that couldn't be matched and removed. Finally, you convert the stack to a string and return it, giving you the reduced string without any adjacent duplicates.
Learn more about Stack patterns.
Solution Approach
The solution to this problem utilizes a commonly used data structure known as a stack. Here's a step-by-step approach to how the algorithm works with the stack to remove duplicates:
- Initialize an empty stack,
stk
. This will be used to keep track of characters in the string as they are processed. - Iterate through each character
c
in the input strings
. - For each character, check if the stack is not empty and if the character at the top of the stack (the last character added to the stack) is equal to the current character
c
.- If yes, that means we have found two adjacent, equal characters. The character at the top of the stack is removed using the
pop()
operation, which effectively removes the pair of adjacent duplicates from our consideration. - If no, there is no pair to remove. In this case, the current character
c
is added to the top of the stack with theappend()
method.
- If yes, that means we have found two adjacent, equal characters. The character at the top of the stack is removed using the
- Repeat this process until all characters from the input string have been examined.
After the iteration is complete:
- The stack now contains the final string with all possible pairs removed. Since a stack is a LIFO (last-in, first-out) structure, to form the final string from the leftover characters, we convert the stack to a string by joining the elements in the stack.
We use join(stk)
to create a string from the elements in the stack.
The implementation in Python using the described approach is efficient because it only processes each character once and uses O(n)
space, where n
is the length of the string (in the worst case when there are no duplicates).
This algorithm is an example of a greedy approach, where we make the local optimal choice at each step (removing adjacent duplicates whenever possible) that leads to the global optimal solution โ the fully reduced string.
Here is the Python code for the reference:
1class Solution:
2 def removeDuplicates(self, s: str) -> str:
3 # Initialize an empty [stack](/problems/stack_intro)
4 stk = []
5 # Iterate through each character in the string
6 for c in s:
7 # Check if the stack is not empty and the top element is equal to the current character
8 if stk and stk[-1] == c:
9 # Remove the last element from the stack i.e., remove duplicate pair
10 stk.pop()
11 else:
12 # Add the current character to the stack
13 stk.append(c)
14 # Convert the stack to a string and return
15 return ''.join(stk)
This solution has a time complexity of O(n)
because each character in the string is processed once. The space complexity is also O(n)
to account for the stack that holds the characters in the process of removing duplicates.
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 use a small example to illustrate the solution approach. Consider the string s = "abbaca"
. We will go through this string and apply the algorithm step by step:
- Initialize an empty stack
stk
. Currently,stk = []
. - Iterate through each character in the string
s
. - The first character is
a
. The stack is empty, so we adda
to the stack. Now,stk = ['a']
. - The second character is
b
. It isn't equal to the top of the stack, which isa
, so addb
to the stack. Now,stk = ['a', 'b']
. - The third character is
b
. It is equal to the top of the stack. Remove the top of the stack, which is the lastb
. Now,stk = ['a']
. - The fourth character is
a
. It equals the top of the stack, so removea
from the stack. Now,stk = []
. - The fifth character is
c
. The stack is empty, so addc
to the stack. Now,stk = ['c']
. - The last character is
a
. It isn't equal to the top of the stack, so adda
to the stack. The stack now looks like this:stk = ['c', 'a']
.
After the iteration is complete, the stack contains ['c', 'a']
. Join these to form the final string, which is "ca"
. This string is the reduced form of the original string s
, with all adjacent duplicates removed. The process and the resulting string are confirmed to be correct as per our algorithm.
The above steps embody the described algorithm to remove pairs of adjacent identical characters from a string and achieve the desired outcome.
Solution Implementation
1class Solution:
2 def removeDuplicates(self, s: str) -> str:
3 # Initialize an empty stack to hold characters.
4 stack = []
5
6 # Loop through each character in the input string.
7 for char in s:
8 # If the stack is not empty and the top element of the stack
9 # is the same as the current character, then we have found
10 # a duplicate and need to remove it from the stack.
11 if stack and stack[-1] == char:
12 stack.pop()
13 else:
14 # If no duplicate found, push the current character onto the stack.
15 stack.append(char)
16
17 # Join all characters left in the stack to form the processed string
18 # without consecutive duplicate characters.
19 return ''.join(stack)
20
1class Solution {
2
3 // Method to remove all adjacent duplicates from the string s.
4 public String removeDuplicates(String s) {
5 // Initialize a StringBuilder to construct the result without duplicates.
6 StringBuilder resultBuilder = new StringBuilder();
7
8 // Iterate over each character of the string.
9 for (char character : s.toCharArray()) {
10 int currentLength = resultBuilder.length();
11 // If the result has characters and the last character is the same as the current one,
12 // we should remove the last character to eliminate the pair of duplicates.
13 if (currentLength > 0 && resultBuilder.charAt(currentLength - 1) == character) {
14 // Delete the last character from resultBuilder.
15 resultBuilder.deleteCharAt(currentLength - 1);
16 } else {
17 // Otherwise, append the current character to resultBuilder.
18 resultBuilder.append(character);
19 }
20 }
21
22 // Convert the StringBuilder back to a String and return the result.
23 return resultBuilder.toString();
24 }
25}
26
1#include <string> // Include string library for string usage
2
3class Solution {
4public:
5 // Method to remove all adjacent duplicates in the string s
6 std::string removeDuplicates(std::string s) {
7 std::string result; // Create an empty string to use as a stack
8
9 // Iterate through each character in the input string
10 for (char character : s) {
11 // If result is not empty and the last character is the same as the current one,
12 // remove the last character from result (simulating a stack pop operation)
13 if (!result.empty() && result.back() == character) {
14 result.pop_back();
15 } else {
16 // Otherwise, add the current character to result (simulating a stack push operation)
17 result.push_back(character);
18 }
19 }
20
21 return result; // Return the resulting string after all duplicates are removed
22 }
23};
24
1/**
2 * Removes all instances of adjacent duplicates in the given string.
3 * @param {string} inputString - The string from which to remove duplicates.
4 * @return {string} The modified string with all adjacent duplicates removed.
5 */
6const removeDuplicates = (inputString: string): string => {
7 // Initialize an empty array to use as a stack.
8 const stack: string[] = [];
9
10 // Iterate through each character of the input string.
11 for (const character of inputString) {
12 // If the stack is not empty and the top element matches the current character...
13 if (stack.length > 0 && stack[stack.length - 1] === character) {
14 // Pop the top element from the stack to remove the duplicate.
15 stack.pop();
16 } else {
17 // If no duplicate is found, push the current character onto the stack.
18 stack.push(character);
19 }
20 }
21
22 // Combine the elements in the stack to form the modified string and return it.
23 return stack.join('');
24};
25
Time and Space Complexity
The given Python code implements a function that removes all adjacent duplicates in a string by using a stack. It iterates through each character in the input string, and removes characters from the stack when a duplicate is detected or appends characters otherwise. Let's analyze the time and space complexities of the code.
Time Complexity
The time complexity is determined by the number of iterations the algorithm performs and the operations executed per iteration. For this code:
- The algorithm iterates through each character of the input string once.
- For each character, the algorithm performs a constant time operation to check if the stack is non-empty and if the top of the stack is equal to the current character.
- In the worst case (when all characters are unique), each character gets added to the stack. In the best case (when all characters are duplicates), each character gets compared and none is added to the stack.
- The
pop
andappend
operations on the stack take constant time, i.e.,O(1)
.
Therefore, the time complexity of the algorithm is O(n)
, where n
is the length of the input string.
Space Complexity
The space complexity considers the additional space used by the algorithm as a function of the input size:
- The stack
stk
is the primary data structure used to store characters from the input string temporarily. - In the worst case scenario, where there are no consecutive duplicate characters, the stack will contain all
n
characters of the input string. - In the best case, where all characters are consecutive duplicates, the stack will be empty at the end of the iteration.
Hence, the space complexity of the algorithm is also O(n)
.
In summary, the function removeDuplicates
has a time complexity of O(n)
and a space complexity of O(n)
.
Learn more about how to find time and space complexity quickly using problem constraints.
A person thinks of a number between 1 and 1000. You may ask any number questions to them, provided that the question can be answered with either "yes" or "no".
What is the minimum number of questions you needed to ask so that you are guaranteed to know the number that the person is thinking?
Recommended Readings
Stack Intro Imagine you have a pile of books on your desk If you want to add a new book you place it on top If you want to read a book you take it from the top And if you simply want to see which book is on the top you
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.