Backtracking 

Prerequisites :

·         Recursion

·         Complexity Analysis

Backtracking is an algorithmic-technique for solving problems recursively by trying to build a solution incrementally, one piece at a time, removing those solutions that fail to satisfy the constraints of the problem at any point of time (by time, here, is referred to the time elapsed till reaching any level of the search tree).

According to the wiki definition,

There are three types of problems in backtracking –

1.      Decision Problem – In this, we search for a feasible solution.

2.      Optimization Problem – In this, we search for the best solution.

3.      Enumeration Problem – In this, we find all feasible solutions.

How to determine if a problem can be solved using Backtracking?

Generally, every constraint satisfaction problem which has clear and well-defined constraints on any objective solution, that incrementally builds candidate to the solution and abandons a candidate (“backtracks”) as soon as it determines that the candidate cannot possibly be completed to a valid solution, can be solved by Backtracking. However, most of the problems that are discussed, can be solved using other known algorithms like Dynamic Programming or Greedy Algorithms in logarithmic, linear, linear-logarithmic time complexity in order of input size, and therefore, outshine the backtracking algorithm in every respect (since backtracking algorithms are generally exponential in both time and space). However, a few problems still remain, that only have backtracking algorithms to solve them until now.

Consider a situation that you have three boxes in front of you and only one of them has a gold coin in it but you do not know which one. So, in order to get the coin, you will have to open all of the boxes one by one. You will first check the first box, if it does not contain the coin, you will have to close it and check the second box and so on until you find the coin. This is what backtracking is, that is solving all sub-problems one by one in order to reach the best possible solution.

Consider the below example to understand the Backtracking approach more formally,

Given an instance of any computational problem  and data  corresponding to the instance, all the constraints that need to be satisfied in order to solve the problem are represented by . A backtracking algorithm will then work as follows:

The Algorithm begins to build up a solution, starting with an empty solution set S = {}

1.      Add to  the first move that is still left (All possible moves are added to  one by one). This now creates a new sub-tree  in the search tree of the algorithm.

2.      Check if  satisfies each of the constraints in .

·         If Yes, then the sub-tree  is “eligible” to add more “children”.

·         Else, the entire sub-tree  is useless, so recurs back to step 1 using argument .

3.      In the event of “eligibility” of the newly formed sub-tree , recurs back to step 1, using argument .

4.      If the check for  returns that it is a solution for the entire data . Output and terminate the program.
If not, then return that no solution is possible with the current 
 and hence discard it.

Pseudo Code for Backtracking :

1.      Recursive backtracking solution.

2.      void findSolutions(n, other params) :

3.         if (found a solution) :

4.             solutionsFound = solutionsFound + 1;

5.             displaySolution();

6.             if (solutionsFound >= solutionTarget) :

7.                 System.exit(0);

8.             return

9.       

10.     for (val = first to last) :

11.         if (isValid(val, n)) :

12.             applyValue(val, n);

13.             findSolutions(n+1, other params);

14.             removeValue(val, n);

15.  Finding whether a solution exists or not

16.  boolean findSolutions(n, other params) :

17.     if (found a solution) :

18.         displaySolution();

19.         return true;

20.   

21.     for (val = first to last) :

22.         if (isValid(val, n)) :

23.             applyValue(val, n);

24.             if (findSolutions(n+1, other params))

25.                 return true;

26.             removeValue(val, n);

27.         return false;

Let us try to solve a standard Backtracking problem, N-Queen Problem.
The N Queen is the problem of placing N chess queens on an N×N chessboard so that no two queens attack each other. For example, following is a solution for 4 Queen problem.

 

The expected output is a binary matrix which has 1s for the blocks where queens are placed. For example, following is the output matrix for the above 4 queen solution.

{ 0,  1,  0,  0}

{ 0,  0,  0,  1}

{ 1,  0,  0,  0}

{ 0,  0,  1,  0}

Backtracking Algorithm: The idea is to place queens one by one in different columns, starting from the leftmost column. When we place a queen in a column, we check for clashes with already placed queens. In the current column, if we find a row for which there is no clash, we mark this row and column as part of the solution. If we do not find such a row due to clashes then we backtrack and return false.

1) Start in the leftmost column

2) If all queens are placed

   return true

3) Try all rows in the current column.  Do following for every tried row.

   a) If the queen can be placed safely in this row then mark this [row,

       column] as part of the solution and recursively check if placing 

       queen here leads to a solution.

   b) If placing the queen in [row, column] leads to a solution then return

       true.

   c) If placing queen doesn't lead to a solution then unmark this [row,

       column] (Backtrack) and go to step (a) to try other rows.

3) If all rows have been tried and nothing worked, return false to trigger

   backtracking.