3.5 Uninformed Search Strategies

The third edition of Artificial Intelligence: foundations of computational agents, Cambridge University Press, 2023 is now available (including full text).

3.5.3 Iterative Deepening

Neither of the preceding methods is ideal. Breadth-first search, which guarantees that a path will be found, requires exponential space. Depth-first search may not halt on infinite graphs or graphs with cycles. One way to combine the space efficiency of depth-first search with the optimality of breadth-first search is to use iterative deepening. The idea is to recompute the elements of the breadth-first frontier rather than storing them. Each recomputation can be a depth-first search, which thus uses less space.

Iterative deepening repeatedly calls a depth-bounded searcher, a depth-first searcher that takes in an integer depth bound and never explores paths with more arcs than this depth bound. Iterative deepening first does a depth-first search to depth 1 by building paths of length 1 in a depth-first manner. If that does not find a solution, it can build paths to depth 2, then depth 3, and so on until a solution is found. When a search with depth-bound n fails to find a solution, it can throw away all of the previous computation and start again with a depth-bound of n+1. Eventually, it will find a solution if one exists, and, as it is enumerating paths in order of the number of arcs, a path with the fewest arcs will always be found first.

To ensure it halts for finite graphs, iterative deepening search needs to distinguish between

  • failure because the depth bound was reached and

  • failure due to exhausting the search space.

In the first case, the search must be retried with a larger depth bound. In the second case, it is a waste of time to try again with a larger depth bound, because no path exists no matter what the depth, and so the whole search should fail.

1: procedure ID_search(G,s,goal)
2:      Inputs
3:          G: graph with nodes N and arcs A
4:          s: start node
5:          goal: Boolean function on nodes      
6:      Output
7:          path from s to a node for which goal is true
8:          or if there is no such path      
9:      Local
10:          hit_depth_bound: Boolean
11:          bound: integer
12:          procedure Depth_bounded_search(n0,,nk,b)
13:               Inputs
14:                   n0,,nk: path
15:                   b: integer, b0               
16:               Output
17:                   path to goal of length k+b if one exists
18:               if b>0 then
19:                   for each arc nk,nA do
20:                        res:=Depth_bounded_search(n0,,nk,n,b-1)
21:                        if res is a path then
22:                            return res                                          
23:               else if goal(nk) then
24:                   return n0,,nk
25:               else if nk has any neighbors then
26:                   hit_depth_bound:=true                             
27:      bound:=0
28:      repeat
29:          hit_depth_bound:=false
30:          res:=Depth_bounded_search(s,bound)
31:          if res is a path then
32:               return res          
33:          bound:=bound+1
34:      until not hit_depth_bound
Figure 3.8: ID_search: iterative deepening search

Pseudocode for iterative deepening search, ID_search, is presented in Figure 3.8. The local procedure Depth_bounded_search implements a depth-bounded depth-first search (using recursion to keep the stack) that places a limit on the length of the paths for which it is searching. It either returns a path, or reaches the end of its code, and returns with no path. It uses a depth-first search to find paths of length k+b, where k is the path length of the given path from the start and b is a non-negative integer. The iterative deepening searcher calls Depth_bounded_search for increasing depth bounds. This program finds the paths to goal nodes in the same order as does breadth-first search. Note that it only needs to check for a goal when b=0, because it knows that there are no solutions for lower bounds.

To ensure that iterative deepening search fails whenever breadth-first search would fail, it needs to keep track of when increasing the bound could help find an answer. A depth-bounded search fails naturally – it fails by exhausting the search space – if the search did not prune any paths due to the depth bound. In this case, the program can stop and report no paths. This is handled through the variable hit_depth_bound in Figure 3.8, which is false when Depth_bounded_search is called initially, and becomes true if the search is pruned due to the depth bound. If it is true at the end of a depth-bounded search, the search failed due to hitting the depth bound, and so the depth bound can be increased, and another depth-bounded search is carried out.

The obvious problem with iterative deepening is the wasted computation that occurs at each step. This, however, may not be as bad as one might think, particularly if the branching factor is high. Consider the running time of the algorithm. Assume a constant branching factor of b>1. Consider the search where the bound is k. At depth k, there are bk nodes; each of these has been generated once. The nodes at depth k-1 have been generated twice, those at depth k-2 have been generated three times, and so on, and the nodes at depth 1 have been generated k times. Thus, the total number of paths expanded is

bk+2bk-1+3bk-2++kb =bk(1+2b-1+3b-2++kb1-k)
<bk(i=1ib(1-i))
=bk(bb-1)2.

Breadth-first search expands i=1kbi=(bk+1-1b-1)=bk(bb-1)-1b-1 nodes. Thus iterative deepening has an asymptotic overhead of b(b-1) times the cost of expanding the nodes at depth k using breadth-first search. Thus, when b=2 there is an overhead factor of 2, and when b=3 there is an overhead of 1.5. This algorithm is O(bk) and there cannot be an asymptotically better uninformed search strategy. Note that, if the branching factor is close to 1, this analysis does not work because then the denominator would be close to zero; see Exercise 9.