Skip to main content

Solving Expression Puzzles with Backtracking

Problem Description

I was asked this question by my friend the other day. Say you are given a mathematically equation in an encoded form. Would it be possible to assign values to characters in an express so that it is valid:

Ex.1:

For expression "ABC" + "ABC" = "DEA"
The assignment {'A': 2, 'B': 3, 'C': 1, 'D': 4, 'E': 6} will make the expression true:
if we subsitute the numbers to their according letters:
the expression becomes 231 + 231 = 462 (which is valid)

The other assignments could be:
432 + 432 = 864
281 + 281 = 562
271 + 271 = 542

Ex.2:

For expression "AA" + "B" = "CDE"
The assignment {'A':9, 'B': 3, 'C': 1, 'D': 0, 'E': 2} will make the expression true:
if we subsitute the numbers to their according letters:
the expression becomes 99 + 3 = 102 (which is valid)

Reduction

Decomposition

This is obviously not an easy problem because there isn't a strict mathematical formula to solve this problem.

However, if we can find a way to validate an assignment, we can use backtracking to enumerate and explore all possible combinations.

Step 1: Find a way to validate the proposed solution

int getNumber(String s, Map<Character, Integer> proposedSolution){
int num = 0;
for(int i = 0; i < s.length(); i++){
char c = s.charAt(i);
num = num * 10 + proposedSolution.get(c);
}
return num;
}

// Checks whether or not the proposedSolution is correct.
// E.g., puzzle={"ONE", "ONE", "TWO"}, proposedSolution={O: 2, N:3, E:1, T:4, W:6} -> true
boolean checkSolution(Map<Character, Integer> proposedSolution) {
int firstNum = getNumber(first, proposedSolution);
int secondNum = getNumber(second, proposedSolution);
int res = getNumber(result, proposedSolution);
return firstNum + secondNum == res;
}

Step 2: Use backtracking to solve the problem

Now, since we have a validation function checkSolution to check if a solution is valid, we have the successful base case for our backtracking algorithm. We can go ahead and implement the solution:

Set<Character> getCharSet(){
Set<Character> set = new HashSet<Character>();
for(int i = 0; i < first.length(); i++){
set.add(first.charAt(i));
}
for(int i = 0; i < second.length(); i++){
set.add(second.charAt(i));
}
for(int i = 0; i < result.length(); i++){
set.add(result.charAt(i));
}
return set;
}

boolean getNextNum(Map<Character, Integer> proposalMap, Set<Integer> usedDigits, char[] charArr, int pos){
if(pos == charArr.length){
if(checkSolution(proposalMap)){
return true;
} else {
return false;
}
}

for(int i = 0; i < 10; i++){
if(usedDigits.contains(i)) continue;
usedDigits.add(i);
proposalMap.put(charArr[pos], i);
if(getNextNum(proposalMap, usedDigits, charArr, pos+1)){
return true;
}
usedDigits.remove(i);
}
return false;
}

Solution

class Puzzle{
String first = ""; // e.g., "ONE"
String second = ""; // e.g., "ONE"
String result = ""; // e.g., "TWO"
Set<Character> set;

public Puzzle(String a, String b, String c){
this.first = a;
this.second = b;
this.result = c;
this.set = getCharSet();
System.out.println(set);
}

int getNumber(String s, Map<Character, Integer> proposedSolution){
int num = 0;
for(int i = 0; i < s.length(); i++){
char c = s.charAt(i);
num = num * 10 + proposedSolution.get(c);
}
return num;
}

// Checks whether or not the proposedSolution is correct.
// E.g., puzzle={"ONE", "ONE", "TWO"}, proposedSolution={O: 2, N:3, E:1, T:4, W:6} -> true
boolean checkSolution(Map<Character, Integer> proposedSolution) {
int firstNum = getNumber(first, proposedSolution);
int secondNum = getNumber(second, proposedSolution);
int res = getNumber(result, proposedSolution);
return firstNum + secondNum == res;
}

Set<Character> getCharSet(){
Set<Character> set = new HashSet<Character>();
for(int i = 0; i < first.length(); i++){
set.add(first.charAt(i));
}
for(int i = 0; i < second.length(); i++){
set.add(second.charAt(i));
}
for(int i = 0; i < result.length(); i++){
set.add(result.charAt(i));
}
return set;
}

boolean getNextNum(Map<Character, Integer> proposalMap, Set<Integer> usedDigits, char[] charArr, int pos){
if(pos == charArr.length){
if(checkSolution(proposalMap)){
return true;
}
else{
return false;
}
}

for(int i = 0; i < 10; i++){
if(usedDigits.contains(i)) continue;
usedDigits.add(i);
proposalMap.put(charArr[pos], i);
if(getNextNum(proposalMap, usedDigits, charArr, pos+1)){
return true;
}
usedDigits.remove(i);
}
return false;
}

Map<Character, Integer> solve() {
Map<Character, Integer> map = new HashMap<Character, Integer>();
Set<Integer> digits = new HashSet<>();
char[] charSet = new char[set.size()];
int pos =0;
for(char c : this.set){
charSet[pos++] = c;
}
if(getNextNum(map, digits, charSet, 0)){
return map;
}
else{
return null;
}
}
}