@cww97
2017-12-21T16:55:06.000000Z
字数 3220
阅读 1378
机器学习
状态下,采取最佳行动:
: Action
为执行行动达到的状态
:对状态的评估
主要有两种方法:
所以问题的重点就是这个评估函数
极小极大搜索(MinMaxSearch)
Value:
其中:
为w处的权值,经验数据???经验数据???
Mobility() 己方行动力大小:定义为可以着子的位置总数
考虑两个人下棋和:
从角度来看,每一步总是使自己获得最大收益,我们称之为过程
从角度来看,每一步要使得自己获得最大收益也就是使获得最小收益,我们称之为过程
在不考虑一方停止的情况下,下棋就是和相互交替的过程
假设搜索的,那么显然会选择令自己收益最大的状态, 则最大收益为, 简称:贪心
假设搜索的, 那么在做出一个move之后,显然会选择令收益最小的状态, 那么先生的最大收益就是,
以此类推,越深,选择会越来越明智.
Input: 棋盘, 搜索深度
Output: 最佳动作
best_val = -INF;
foreach move do
MakeMove(board, move, my_color); //自己做该move
val = Min(board, depth - 1); //对方找一个Min
UnMakeMove(board, move, my_color);
if val > best_val: // 我们需要最大化这个Min
best_val = val;
best_move = move;
return best_move
Input: 棋盘, 搜索深度
Output: 子节点最小值
ans = INF
if (depth < 0) return Value(board); // Value为估价函数
if (!CanMove(board, opp_color)):
if (!CanMove(board, my_color)):
return Value(board); // Game Over
return Max(board, depth); // 白棋停一回合,黑棋走
// 白子可以move
foreach move do:
MakeMove(board, move, opp_color); // 白棋做一个move
val = Max(board, depth - 1); // 黑棋找一个Max
UnMakeMove(board, move, opp_color);
ans = min(ans, val); // 白棋需要最小化这个最大值 from Black
return ans;
Input: 棋盘, 搜索深度
Output: 子节点最大值
ans = -INF
if (depth < 0) return Value(board); // Value为估价函数
if (!CanMove(board, opp_color)):
if (!CanMove(board, my_color)):
return Value(board); // Game Over
return Min(board, depth); //黑棋停一回合,白棋走
// 黑子可以move
foreach move do:
MakeMove(board, move, opp_color); // 黑棋做一个move
val = Min(board, depth - 1); // 白棋找一个Min
UnMakeMove(board, move, opp_color);
ans = max(ans, val); // 白棋需要最小化这个最大值 from Black
return ans;
这个搜算算法逻辑很简单易懂,不过,为了编写代码的方便Min和Max的过程可以合并
如下NegaMax
:
Input:棋盘 board, 移动的一方color,搜索深度depth
Output:子节点的最大值
ans = -INF
sign[my_color] = 1;
sign[opp_color] = -1;
if (depth <= 0) return sign[color] * Value(board);
if (!canMove(board, color)):
if (!canMove(board, color^1)):
return sign[color] * Value(board);
return - NegaMax(board, color^1, depth);
foreach move do:
MakeMove(board, move, color);
val = -NegaMax(board, color^1, depth-1);
UnMakeMove(board, move, color);
ans = max(ans, val);
return ans;
先看个栗子🌰
A
是Max
节点,取B
、C
、D
的最大值,dfs(A)
,我们首先会调用Min(B)
, 得到B
的值为6
,随后Min(C)
,在这过程中我们先Max(E) = 4
, 到现在我们可以发现:因为C
是Min
节点,所以,M(C) <= 4
, 而因为A
是Max
节点,搜索过B
之后我们得到的结论是Max(A) >= 6
, 至此,我们得到结论:C
是个没有用的节点,那么对F
和G
的搜索也是没有必要的,下一步可以直接搜索D
节点...
剪枝,剪枝,剪枝!!!
为了利用这个性质,我们决定引入两个值: 和 , 分别表示节点估值的下界和上界,
Input: 棋盘 board, alpha , beta (为方便代码中用a和b表示),移动的一方 color,搜索深度depth
Output: 子节点最大值
ans = -INF;
sign[my_color] = 1;
sign[opp_color] = -1;
if (depth <= 0) return sign[color] * Value(board);
if (!canMove(board, color)):
if (!canMove(board, color^1)):
return sign[color] * Value(board);
return - AlphaBeta(board, -b, -a, color^1, depth);
foreach move do:
MakeMove(board, move, color);
val = -AlphaBeta(board, -b, -a, -1, color^1);
UnmakeMove(board, move, color);
if (val > a && val >= b) return val;
a = max(a, val);
return ans;
AlphaBeta搜索的节点大致可以分为三类(以Max),过程为例:
我们可以估算价值然后进行排序来改变搜索顺序,使得剪枝的节点尽量多
and,一个棋盘状态可以在双方不同的走法中走到,但是这个实际上只需要搜索一次,我们可以记录他的值,下一次访问到的时候直接return,其实是一个记忆化的过程