282. Expression Add Operators - Detailed Explanation
Problem Statement
You are given a string num that contains only digits and an integer target. Your task is to return all valid expressions (as strings) that can be built by inserting the binary operators '+'
, '-'
, or '*'
between the digits in num so that the resulting expression evaluates to the target value. Note that the digits must remain in their original order and you cannot insert any extra characters or change the order of the digits.
Important Points:
-
Operands: Formed by combining consecutive digits.
-
No Leading Zeros: Operands cannot have extra leading zeros (except the number zero itself).
-
Operator Precedence: The multiplication operator (
*
) takes precedence over addition (+
) and subtraction (-
).
Example 1
- Input:
num = "123" target = 6
- Output:
["1+2+3", "1*2*3"]
- Explanation:
Both expressions evaluate to 6:1+2+3 = 6
1*2*3 = 6
Example 2
- Input:
num = "232" target = 8
- Output:
["2*3+2", "2+3*2"]
- Explanation:
Both expressions evaluate to 8:2*3+2 = 6+2 = 8
2+3*2 = 2+6 = 8
Example 3
- Input:
num = "105" target = 5
- Output:
["1*0+5", "10-5"]
- Explanation:
1*0+5 = 0+5 = 5
10-5 = 5
Note that"1*05"
is not allowed because the operand"05"
has a leading zero.
Constraints
- 1 <= num.length <= 10
- num consists of only digits.
- -2³¹ <= target <= 2³¹ - 1
Hints for the Approach
-
Hint 1: Use backtracking (or DFS) to try every possible way to split the string into numbers and insert operators between them. This means at each step, decide the length of the current number and then try all three operators (or no operator at the very beginning).
-
Hint 2: Think about handling the multiplication operator carefully. Since multiplication has a higher precedence than addition and subtraction, maintain an extra variable (e.g.,
tail
ormulted
) that keeps track of the last operand so you can correctly update the current evaluation when a multiplication is encountered.
Approach 1: Brute Force Generation and Evaluation
Idea
-
Generate All Expressions:
Iterate over all possible ways to split the input string and insert operators. For every possible expression, evaluate it and check if it equals the target. -
Evaluation:
Use a helper function to parse and evaluate the generated string expression.
Drawbacks
-
Efficiency:
Generating and evaluating every possible expression without pruning will be extremely inefficient, especially when the string length grows, because the number of possibilities grows exponentially. -
Operator Precedence:
Handling the multiplication operator correctly (considering operator precedence) makes evaluation tricky.
Approach 2: Backtracking with Recursive DFS (Optimal)
Idea
Instead of generating all expressions first, use a recursive backtracking approach to build the expression on the fly while keeping track of the current evaluation. For each recursive call, decide:
-
Partition:
How many digits to take from the current index to form the next operand. -
Operators:
If it’s not the first operand, choose to insert'+'
,'-'
, or'*'
before the operand. -
Tracking Values:
Maintain three values:expr
: The current expression string.calc
: The current calculated value of the expression.tail
: The value of the last operand (used to handle multiplication).
Handling Multiplication
- When adding
'+'
or'-'
, updatecalc
normally. - For
'*'
, subtract the last operand (tail) fromcalc
and add the result oftail * current_number
to handle the multiplication precedence
Pseudocode Overview
function backtrack(index, expr, calc, tail):
if index == length of num:
if calc == target:
add expr to result list
return
for i from index to length of num:
if number formed has leading zero (and length > 1): break
current = num[index...i]
if index == 0:
backtrack(i+1, current, current, current)
else:
backtrack(i+1, expr + "+" + current, calc + current, current)
backtrack(i+1, expr + "-" + current, calc - current, -current)
backtrack(i+1, expr + "*" + current, calc - tail + tail * current, tail * current)
Detailed Explanation of the DFS Approach
Step-by-Step Walkthrough
-
Initialization:
Start from index 0 with an empty expression string, a calculated value of 0, and no tail value. -
Recursive Decision Making:
For every recursive call:- Choose the next number by iterating from the current index to the end of the string.
- Skip numbers with leading zeros (if the chosen number has more than one digit and starts with
'0'
).
-
Handling the First Number:
If it is the first number (i.e., index is 0), add it to the expression without any operator. Set bothcalc
andtail
to this number. -
Adding Operators:
For subsequent numbers, try all three possible operators:- Addition (
+
):
Updatecalc
ascalc + current
and settail
tocurrent
. - Subtraction (
-
):
Updatecalc
ascalc - current
and settail
to-current
. - Multiplication (
*
):
Since multiplication has a higher precedence, updatecalc
ascalc - tail + (tail * current)
and updatetail
totail * current
.
- Addition (
-
Base Case:
When the index reaches the end of num, check if the computedcalc
equals target. If so, add the expression to the result list.
Visual Example
Consider num = "123"
and target = 6
:
- First Call:
- Choose "1" (index 0 to 0), set
expr = "1"
,calc = 1
,tail = 1
.
- Choose "1" (index 0 to 0), set
- Second Call (index = 1):
- Choose "2" (index 1 to 1) and try:
"1+2"
:calc = 1 + 2 = 3
,tail = 2
"1-2"
:calc = 1 - 2 = -1
,tail = -2
"1*2"
:calc = 1 - 1 + (1*2) = 2
,tail = 2
- Choose "2" (index 1 to 1) and try:
- Third Call (index = 2):
For each branch from above, choose "3":- From
"1+2"
:"1+2+3"
:calc = 3 + 3 = 6
"1+2-3"
:calc = 3 - 3 = 0
"1+2*3"
:calc = 3 - 2 + (2*3) = 7
- From
"1*2"
:"1*2+3"
:calc = 2 + 3 = 5
"1*2-3"
:calc = 2 - 3 = -1
"1*2*3"
:calc = 2 - 2 + (2*3) = 6
- From
- Result:
The valid expressions are"1+2+3"
and"1*2*3"
.
Python Implementation
Java Implementation
Complexity Analysis
-
Time Complexity:
The number of recursive calls is exponential in the length of num since at every index we try every possible split and operator. In the worst case, the time complexity is roughly (O(4^n)) (each recursive call can branch up to 4 ways – one for the case with no operator at the start and three for each operator insertion). -
Space Complexity:
The space complexity is (O(n)) due to the recursion stack and the space required to build the expression strings.
Common Mistakes
-
Ignoring Leading Zeros:
Not checking for numbers with leading zeros can lead to invalid expressions (e.g.,"05"
). -
Incorrect Multiplication Handling:
Failing to adjust the calculated value when using multiplication leads to incorrect evaluation due to operator precedence. -
Not Considering the Entire String:
Ensure that every recursive call eventually consumes the entire string and checks the computed value against the target.
Edge Cases
-
Single-Digit Input:
When num is only one digit, the only valid expression is the digit itself. -
All Zeros:
For example,"000"
with target0
might produce multiple expressions like"0+0+0"
,"0*0-0"
, etc. Make sure to handle duplicates appropriately if needed. -
Large Numbers:
The input string might form large numbers. Using a data type (likelong
in Java) for intermediate calculations can help avoid overflow.
Alternative Variations
-
Allowing Parentheses:
How would the solution change if you could add parentheses to change the order of operations? -
Different Operators:
What if the problem included division or other operators? -
Evaluation of Boolean Expressions:
Similar techniques can be used to form expressions that evaluate to a Boolean value.
Related Problems for Further Practice
-
Letter Combinations of a Phone Number:
Practice generating combinations from a string. -
Restore IP Addresses:
Use backtracking to form valid IP addresses from a string. -
Generate Parentheses:
Use DFS/backtracking to generate all valid combinations of parentheses. -
Target Sum:
A problem that involves adding operators (or choices) to achieve a target sum.
GET YOUR FREE
Coding Questions Catalog
