Artificial Intelligence - Heuristic/Local search
Heuristic Function
• A heuristic function h(n) yields the estimated cost of the cheapest path from the state at node n to a goal state.
• What do we mean by “heuristic”?
- Oxford Dictionary: Proceeding to a solution by trial and error or by rules that are only loosely defined.
- Wikipedia: A technique designed to solve a problem that ignores whether the solution can be proven to be correct, but which usually produces a good solution or solves a simpler problem that contains or intersects with the solution of the more complex problem.
- In this post - ex) Greedy Best-First Search and A*
Search Terminology
data:image/s3,"s3://crabby-images/e9354/e9354eb2cebcf1a98400c8272837e48686a32fef" alt=""
- Tree Search
1 2 3 4 5 6 7 8 9 10 | frontier.add(INITIAL_STATE) while not is_empty(frontier): node = frontier.get_next_node() for nd in expand(node): if is_goal(nd): return solution elif (nd not in frontier): frontier.add(nd) return failure | cs |
data:image/s3,"s3://crabby-images/64f24/64f2451fe61265ef5bc3c704386e7e9555b08143" alt=""
- Graph Search
1 2 3 4 5 6 7 8 9 10 11 12 | frontier.add(INITIAL_STATE) explored = {} while not is_empty(frontier): node = frontier.get_next_node() explored.add(node) for nd in expand(node): if is_goal(nd): return solution elif (nd not in frontier) and (nd not in explored): frontier.add(nd) return failure | cs |
data:image/s3,"s3://crabby-images/a9c44/a9c441b6aa4e070ce320575acb744c4238b455d1" alt=""
-
Tree Search vs. Graph Search
- Without some sort of bookkeeping, naive tree search methods may repeat themselves, possibly ad infinitum
- Graph search methods maintain a data structure to track previously explored states, avoiding repetition
- Graph search is going to take up more space given the extra data structure
- For our purposes, we’ll always assume that we’re using graph search unless explicitly said otherwise
- Breadth-First Search (BFS)
1 2 3 4 5 6 7 8 9 10 11 12 | frontier.add(INITIAL_STATE) # FIFO queue explored = {} while not is_empty(frontier): node = frontier.pop() # dequeue explored.add(node) for nd in expand(node): if is_goal(nd): return solution elif (nd not in frontier) and (nd not in explored): frontier.add(nd) return failure | cs |
data:image/s3,"s3://crabby-images/15425/1542581faeec3b5c86565fbd2a5d87bc448ce4a1" alt=""
- Complete? Yes (if branching factor b is finite)
- Optimal? Yes, if step costs are identical, but not in general
- Time? 1+b+b2+b3+…+bd = O(bd)
- Space? O(bd) (b = branching factor, d = depth of solution)
data:image/s3,"s3://crabby-images/c0c17/c0c1714aff075db44382e953cb03e97aff7c6af6" alt=""
- Depth-First Search (DFS)
1 2 3 4 5 6 7 8 9 10 11 12 | frontier.add(INITIAL_STATE) # LIFO stack explored = {} while not is_empty(frontier): node = frontier.pop() explored.add(node) for nd in expand(node): if is_goal(nd): return solution elif (nd not in frontier) and (nd not in explored): frontier.add(nd) return failure | cs |
data:image/s3,"s3://crabby-images/62654/626545b368675192803438eaad6aa72abc81d888" alt=""
DFS ( Graph )
1 2 3 4 5 6 7 8 9 10 11 12 13 | frontier.add(INITIAL_STATE) # LIFO stack explored = {} while not is_empty(frontier): node = frontier.pop() explored.add(node) for n in expand(node): if (n not in frontier) and (n not in explored): if is_goal(n): return solution else: frontier.add(n) return failure | cs |
DFS ( Tree )
1 2 3 4 5 6 7 8 9 10 11 12 13 | frontier.add(INITIAL_STATE) explored = {} while not is_empty(frontier): node = frontier.pop() for n in expand(node): if (n not in frontier): if is_goal(n): return solution else: frontier.add(n) return failure | cs |
data:image/s3,"s3://crabby-images/87cc2/87cc29156ce8580a8d1509c7fce551e10b7a97fa" alt=""
- Uniform-Cost Search
• How can we modify BFS to retain optimality in problems with different transition costs?
• Intuition: expand nodes in order of least path cost so far
• Instead of FIFO or LIFO queue, store frontier as a priority queue
• Recall: priority queues are an abstract data type that holds a collection of elements associated with a (usually numeric) priority, such that higher priority items are dequeued (popped) before lower ones.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | frontier.add(INITIAL_STATE) # Priority queue explored = {} while not is_empty(frontier): node = frontier.pop() if is_goal(node): return solution explored.add(node) for nd in expand(node): if (nd not in frontier) and (nd not in explored): frontier.add(nd) else if (nd in frontier) and (frontier.get(nd) > nd.cost): frontier.update(nd, nd.cost) return failure | cs |
data:image/s3,"s3://crabby-images/42c6d/42c6d74249935bdc9788d6756c7828df6468964c" alt=""
- Complete? Yes
- Optimal? Yes
- Time? O(b^(1+[C*/ϵ]))
- Space? O(b^(1+[C*/ϵ]))
(b = branching factor, C* = cost of optimal solution, ϵ = minimum step cost)
- Greedy Best-First Search
data:image/s3,"s3://crabby-images/c9a78/c9a7839ef9aefde1d8b7ca0bfcb322af9c64bc46" alt=""
data:image/s3,"s3://crabby-images/d0891/d0891a7c6fda063f45ff58fcb413d6725527d923" alt=""
- A* Search
- Intuition: avoid expanding paths that are already expensive, but still prefer those that seem closest to the goal
- Evaluation function:
- f(n) = g(n) + h(n)
- g(n) = cost so far to reach state n (like UCS)
- h(n) = est. cost of cheapest path from n to goal (like Greedy)
- f(n) = est. cost of cheapest path through n to goal
1 2 3 4 5 6 7 8 9 10 11 12 13 14 | frontier.add(START_STATE, 0) explored = {} while not is_empty(frontier): node = frontier.pop() if is_goal(node): return solution explored.add(node) for nd in expand(node): if (nd not in frontier) and (nd not in explored): frontier.add(nd) else if (nd in frontier) and (frontier.get(nd) > nd.cost): frontier.update(nd, nd.cost) return failure | cs |
data:image/s3,"s3://crabby-images/8a1dd/8a1dd3259600ce00ba14f51048f3be0b787d9cb3" alt=""
data:image/s3,"s3://crabby-images/b9a60/b9a606a241f56a52f3a5ebae8a92248f17217aa2" alt=""
Local Search
• Local search algorithms that keep a single current state, and then try to improve it.
• State space is the set of complete configurations
• In many optimization problems, the path to the goal is irrelevant; the goal state itself is the solution
• We may not know what the goal is! In that case, the search becomes an optimization problem — we return the best state we can find.
data:image/s3,"s3://crabby-images/4bcc9/4bcc96a8ef8142065e5ddf5fa602569c79a18748" alt=""
data:image/s3,"s3://crabby-images/76dd2/76dd2ce3882c0ee432a56a74d31d6df60c052c73" alt=""
- Hill Climbing Search
1 2 3 4 5 6 7 8 | current_node = INITIAL_STATE while not is_goal(current_node): successors = expand(current_node) best = select_best(successors) if objective(current_node) > objective(best): return current_node else: current_node = best | cs |
= 8-Puzzle Hill Climbing
data:image/s3,"s3://crabby-images/886d0/886d09052e0cb118ad3737a2c9652b382ace048f" alt=""
data:image/s3,"s3://crabby-images/82a69/82a697ec5e5490ecd2a39255580dab5af699fe1c" alt=""
Challenges for Hill Climbing
• Local Maxima/Minima – every move has a worse result than the current state. Once a local maximum is reached, there is no way to backtrack or move out of that maximum.
• “Shoulders/Plateaux” – there are no better moves, but some are just as good. Do we keep exploring? Hill climbing can have difficult time finding its way off of a flat portion of the objective function landscape.
So why use local search?
• Low memory requirements – usually constant
• Effective – Can often find good solutions in extremely large state spaces
• Can be useful for optimization problems with continuous, infinite branching factors
• Randomized variants of hill climbing can mitigate many of the drawbacks in practice
data:image/s3,"s3://crabby-images/0993b/0993ba1a3835618e8f3fd4e6a998db2cc3469c90" alt=""
data:image/s3,"s3://crabby-images/5372a/5372a861aea694abd3bebf34b312ac1c9b3bfc77" alt=""
Reference: Matthew Hale Rattigan