2020-09-25 21:16:05 +08:00
|
|
|
from __future__ import annotations
|
2020-09-26 00:08:57 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
import math
|
2019-07-07 12:59:58 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
""" Minimax helps to achieve maximum score in a game by checking all possible moves
|
2020-05-22 14:10:11 +08:00
|
|
|
depth is current depth in game tree.
|
2019-07-07 12:59:58 +08:00
|
|
|
nodeIndex is index of current node in scores[].
|
|
|
|
if move is of maximizer return true else false
|
2020-05-22 14:10:11 +08:00
|
|
|
leaves of game tree is stored in scores[]
|
2019-07-07 12:59:58 +08:00
|
|
|
height is maximum height of Game tree
|
2019-10-05 13:14:13 +08:00
|
|
|
"""
|
2019-07-07 12:59:58 +08:00
|
|
|
|
|
|
|
|
2020-09-26 00:08:57 +08:00
|
|
|
def minimax(
|
|
|
|
depth: int, node_index: int, is_max: bool, scores: list[int], height: float
|
|
|
|
) -> int:
|
2020-09-25 21:16:05 +08:00
|
|
|
"""
|
|
|
|
>>> import math
|
|
|
|
>>> scores = [90, 23, 6, 33, 21, 65, 123, 34423]
|
|
|
|
>>> height = math.log(len(scores), 2)
|
|
|
|
>>> minimax(0, 0, True, scores, height)
|
|
|
|
65
|
|
|
|
>>> minimax(-1, 0, True, scores, height)
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
|
|
|
ValueError: Depth cannot be less than 0
|
|
|
|
>>> minimax(0, 0, True, [], 2)
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
|
|
|
ValueError: Scores cannot be empty
|
|
|
|
>>> scores = [3, 5, 2, 9, 12, 5, 23, 23]
|
|
|
|
>>> height = math.log(len(scores), 2)
|
|
|
|
>>> minimax(0, 0, True, scores, height)
|
|
|
|
12
|
|
|
|
>>> minimax('1', 2, True, [], 2 )
|
|
|
|
Traceback (most recent call last):
|
|
|
|
...
|
|
|
|
TypeError: '<' not supported between instances of 'str' and 'int'
|
|
|
|
"""
|
2019-07-07 12:59:58 +08:00
|
|
|
|
2020-09-25 21:16:05 +08:00
|
|
|
if depth < 0:
|
|
|
|
raise ValueError("Depth cannot be less than 0")
|
2019-07-07 12:59:58 +08:00
|
|
|
|
2020-09-25 21:16:05 +08:00
|
|
|
if len(scores) == 0:
|
|
|
|
raise ValueError("Scores cannot be empty")
|
|
|
|
|
|
|
|
if depth == height:
|
|
|
|
return scores[node_index]
|
|
|
|
|
|
|
|
if is_max:
|
2019-10-05 13:14:13 +08:00
|
|
|
return max(
|
2020-09-25 21:16:05 +08:00
|
|
|
minimax(depth + 1, node_index * 2, False, scores, height),
|
|
|
|
minimax(depth + 1, node_index * 2 + 1, False, scores, height),
|
2019-10-05 13:14:13 +08:00
|
|
|
)
|
2020-09-25 21:16:05 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
return min(
|
2020-09-25 21:16:05 +08:00
|
|
|
minimax(depth + 1, node_index * 2, True, scores, height),
|
|
|
|
minimax(depth + 1, node_index * 2 + 1, True, scores, height),
|
2019-10-05 13:14:13 +08:00
|
|
|
)
|
2019-07-07 12:59:58 +08:00
|
|
|
|
2019-10-05 13:14:13 +08:00
|
|
|
|
2020-09-25 21:16:05 +08:00
|
|
|
def main():
|
2019-10-05 13:14:13 +08:00
|
|
|
scores = [90, 23, 6, 33, 21, 65, 123, 34423]
|
|
|
|
height = math.log(len(scores), 2)
|
|
|
|
print("Optimal value : ", end="")
|
|
|
|
print(minimax(0, 0, True, scores, height))
|
2020-09-25 21:16:05 +08:00
|
|
|
|
|
|
|
|
|
|
|
if __name__ == "__main__":
|
|
|
|
import doctest
|
|
|
|
|
|
|
|
doctest.testmod()
|
|
|
|
main()
|