AI Algorithms and Programs
AI Algorithms and Programs
AI Algorithms and Programs
1. To implement A* algorithm
Begin by initializing two sets: the open set and the closed set.
The open set initially contains only the starting node, while the closed set is empty.
Set the cost of reaching the starting node (g-score) to zero and calculate the heuristic cost
estimate to the goal (h-score) for the starting node.
At each iteration of the loop, select the node from the open set with the lowest f-score (f = g +
h).
This node is the most promising candidate for evaluation, as it balances the actual cost incurred
(g) and the estimated remaining cost (h).
For the selected node, consider its neighboring nodes (also known as successors).
Calculate the actual cost to reach each neighbor from the current node (g-score).
Calculate the heuristic cost estimate from each neighbor to the goal (h-score).
For each neighbor, calculate the total estimated cost (f-score) by summing the actual cost (g-
score) and the heuristic estimate (h-score).
If a neighbor is not in the open set, add it to the open set.
If a neighbor is already in the open set and its f-score is lower than the previously recorded f-
score, update the neighbor's f-score and set its parent to the current node. This means a shorter
path to the neighbor has been discovered.
After evaluating the neighbors of the current node, move the current node to the closed set,
indicating that it has been fully evaluated.
Return to the main loop and select the next node for evaluation based on its f-score.
If the goal node is reached, the algorithm terminates, and the optimal path can be
reconstructed by backtracking from the goal node to the starting node using the parent
pointers.
If the open set becomes empty without reaching the goal, the algorithm terminates with the
conclusion that no path exists.
Once the goal is reached, you can reconstruct the optimal path by following the parent pointers
from the goal node back to the starting node. This path represents the shortest route.
A* PROGRAM
AO* ALGORITHM
Like A* algorithm here we will use two arrays and one heuristic function.
OPEN: It contains the nodes that have been traversed but yet not been marked solvable or unsolvable.
h(n): The distance from the current node to the goal node.
--------------------------
Step 3: Select a node n that is both on OPEN and a member of T0. Remove it from OPEN and place it in
CLOSE
Step 4: If n is the terminal goal node then leveled n as solved and leveled all the ancestors of n as solved.
If the starting node is marked as solved then success and exit.
Step 5: If n is not a solvable node, then mark n as unsolvable. If starting node is marked as unsolvable,
then return failure and exit.
Step 6: Expand n. Find all its successors and find their h (n) value, push them into OPEN.
Step 8: Exit.
AO* PROGRAM
# Cost to find the AND and OR path
def Cost(H, condition, weight = 1):
cost = {}
if 'AND' in condition:
AND_nodes = condition['AND']
Path_A = ' AND '.join(AND_nodes)
PathA = sum(H[node]+weight for node in AND_nodes)
cost[Path_A] = PathA
if 'OR' in condition:
OR_nodes = condition['OR']
Path_B =' OR '.join(OR_nodes)
PathB = min(H[node]+weight for node in OR_nodes)
cost[Path_B] = PathB
return cost
Start =Next[0]
Path += '<--' +shortest_path(Start, Updated_cost, H)
# ADD TO PATH FOR AND PATH
else:
Path +='<--('+key[Index]+') '
Start = Next[0]
Path += '[' +shortest_path(Start, Updated_cost, H) + ' + '
Start = Next[-1]
Path += shortest_path(Start, Updated_cost, H) + ']'
return Path
Conditions = {
'A': {'OR': ['B'], 'AND': ['C', 'D']},
'B': {'OR': ['E', 'F']},
'C': {'OR': ['G'], 'AND': ['H', 'I']},
'D': {'OR': ['J']}
}
# weight
weight = 1
# Updated cost
print('Updated Cost :')
Updated_cost = update_cost(H, Conditions, weight=1)
print('*'*75)
print('Shortest Path :\n',shortest_path('A', Updated_cost,H))
BREADTH FIRST SEARCH ALGORITHM
Breadth-First Search uses a queue data structure technique to store the vertices. And the queue follows
the First In First Out (FIFO) principle, which means that the neighbors of the node will be displayed,
beginning with the node that was put first.
The transverse of the BFS algorithm is approaching the nodes in two ways.
Visited node
Not visited node
The steps involved in the BFS algorithm to explore a graph are given as follows –
Step 2: Add that node at the front of the queue to the visited list.
Step 3: Make a list of the nodes as visited that are close to that vertex
Step 6: Exit
graph = {
'5' : ['3','7'],
'3' : ['2', '4'],
'7' : ['8'],
'2' : [],
'4' : ['8'],
'8' : []
}
# Driver Code
print("Following is the Breadth-First Search")
bfs(visited, graph, '5') # function calling
DEPTH FIRST SEARCH ALGORITHM
The outcome of a DFS traversal of a graph is a spanning tree. A spanning tree is a graph that is devoid of
loops. To implement DFS traversal, you need to utilize a stack data structure with a maximum size equal
to the total number of vertices in the graph.
Step 1: Create a stack with the total number of vertices in the graph as the size.
Step 2: Choose any vertex as the traversal's beginning point. Push a visit to that vertex and add it to the
stack.
Step 3: Push any non-visited adjacent vertices of a vertex at the top of the stack to the top of the stack.
Step 4: Repeat steps 3 and 4 until there are no more vertices to visit from the vertex at the top of the
stack.
Step 5: If there are no new vertices to visit, go back and pop one from the stack using backtracking.
Step 7: When the stack is entirely unoccupied, create the final spanning tree by deleting the graph's
unused edges.
graph = {
'5' : ['3','7'],
'3' : ['2', '4'],
'7' : ['8'],
'2' : [],
'4' : ['8'],
'8' : []
}
# Driver Code
print("Following is the Depth-First Search")
dfs(visited, graph, '5')
WATER JUG PROBLEM ALGORITHM
Step 1. Initialize two variables j1 and j2 to represent the current amount of water in each jug.
Step 3. Create a list of possible actions, including filling a jug, emptying a jug, and pouring water from
one jug to the other.
Step 5. Create a stack to keep track of states to visit, and add the initial state to the stack.
Step 7. If the stack becomes empty without finding a solution, return None.
print("Steps: ")
Output:
Steps:
0 0
4 0
4 3
0 3
3 0
3 3
4 2
0 2
HILL CLIMBING ALGORITHM
Simple hill climbing is the simplest way to implement a hill climbing algorithm. It only evaluates the
neighbor node state at a time and selects the first one which optimizes current cost and set it as a current
state. It only checks it's one successor state, and if it finds better than the current state, then move else
be in the same state. This algorithm has the following features:
---------------------------------------------------------------------------------------------------
Step 1: Evaluate the initial state, if it is goal state then return success and Stop.
Step 2: Loop Until a solution is found or there is no new operator left to apply.
Step 5: Exit.
#generate neighbors of the random solution by swapping cities and returns the
best neighbor
def neighbors(matrix, solution):
neighbors = []
for i in range(len(solution)):
for j in range(i + 1, len(solution)):
neighbor = solution.copy()
neighbor[i] = solution[j]
neighbor[j] = solution[i]
neighbors.append(neighbor)
#assume that the first neighbor in the list is the best neighbor
best_neighbor = neighbors[0]
best_path = path_length(matrix, best_neighbor)
def hill_climbing(coordinate):
matrix = generate_matrix(coordinate)
current_solution = solution(matrix)
current_path = path_length(matrix, current_solution)
neighbor = neighbors(matrix,current_solution)[0]
best_neighbor, best_neighbor_path = neighbors(matrix, neighbor)
OUTPUT
The solution is
[2, 4, 8, 3, 7, 5, 0, 6, 9, 10, 1]
MINMAX ALGORITHM
The Minimax Algorithm forms a tree-like structure through recursion, where the two players, Min and
Max take the following steps to reach the final goal state of the game.
Step 2. Evaluate the score of each branch, using the evaluation function.
Selecting the child node with a maximum score for Max player.
Selecting the child node with the minimum score for Min player.
It is based on these values/scores that the quality (good or bad) of the move is decided.
Step 4. At the root node, the node with the max value is selected and the corresponding move is
performed.
Throughout the process, the aim is to minimize the maximum loss or the worst case scenario and get the
best solution for successful game playing
if (maxTurn):
return max(minimax(curDepth + 1, nodeIndex * 2,
False, scores, targetDepth),
minimax(curDepth + 1, nodeIndex * 2 + 1,
False, scores, targetDepth))
else:
return min(minimax(curDepth + 1, nodeIndex * 2,
True, scores, targetDepth),
minimax(curDepth + 1, nodeIndex * 2 + 1,
True, scores, targetDepth))
# Driver code
scores = [3, 5, 2, 9, 12, 5, 23, 23]
treeDepth = math.log(len(scores), 2)