算法笔记--极大极小搜索及alpha-beta剪枝
参考1:https://www.zhihu.com/question/27221568
参考2:https://blog.csdn.net/hzk_cpp/article/details/79275772
参考3:https://blog.csdn.net/BIT1120172185/article/details/80963609
极小极大搜索算法即minimax搜索算法
主要应用于零和博弈(非胜即负,如围棋,象棋,井子棋等),完全信息(玩家知道之前所有的步骤。象棋就是完全信息,因为玩家是交替着落子,且之前的步骤都能在棋盘上体现)
这个算法采用搜索算法递归实现,一层为先手,记为a, 一层为后手,记为b, 交替出现
对于最终局面,有一个分数(比如:先手胜分数为1, 平局分数为0,先手输分数为-1)
先手a想要让这个分数越大越好,后手b想要让这个分数越小越好,于是搜索到先手这一层,取最大返回,搜索到后手这一层,取最小返回
如下图:

于是伪代码为:
int MaxMin(position,int d)
{
int bestvalue,value;
if(game over) //检查游戏是否结束
return evaluation(p);// 游戏结束,返回估值
if(depth<=) //检查是否是叶子节点
return evaluation(p);//叶子节点,返回估值
if(max) //极大值点
bestvalue=-INFINTY;
else //极小值点
bestvalue=INFINTY;
for(each possibly move m)
{
MakeMove(m); //走棋
value=MaxMin(p,d-);
UnMakeMove(m); //恢复当前局面
if(max)
bestvalue=max(value,bestvalue);//取最大值
else
bestvalue=min(value,bestvalue);//取最小值
}
return bestvalue;
}
关于alpha-beta剪枝:
如果当前层为取最小,如果取最小后比上一层当前最大值还小,则不需要往下搜索,因为上一层根本不会选择当前节点往下搜,还有更好的选择
同理,如果当前层为取最大,如果取最大后比上一层当前最小值还大,则不需要往下搜索,因为上一层根本不会选择当前节点往下搜
具体推荐看最上面的知乎链接点赞最多的回答。
alpha-beta剪枝后的伪代码:
int AlphaBeta(nPlay,nAlpha,nBeta)
{
if(game over)
return Eveluation; //胜负已分,返回估值
if(nPly==)
return Eveluation; //叶子节点返回估值
if(Is Min Node) //判断 节点类型
{ // 极小值节点
for(each possible move m)
{
make move m; //生成新节点
score=AlphaBeta(nPly-,nAlpha,nBeta)//递归搜索子节点
unmake move m;//撤销搜索过的节点
if(score<nBeta)
{
nBeta=score;//取极小值
if(nAlpha>=nBeta)
return nAlpha;//alpha剪枝,抛弃后继节点
}
}
return nBeta;//返回最小值
}
else
{//取极大值的节点
for(each possible move m)
{
make move m; //生成新节点
score=AlphaBeta(nPly-,nAlpha,nBeta)//递归搜索子节点
unmake move m;//撤销搜索过的节点
if(score>nAlpha)
{
nAlpha=score;//取极小值
if(nAlpha>=nBeta)
return nBeta;//nBeta剪枝,抛弃后继节点
}
}
return nAlpha;//返回最小值
}
}
例题1:POJ - 1568
思路:井子棋下的步数小于等于4必平局
代码:
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize(4)
#include<cstdio>
#include<iostream>
using namespace std;
#define fi first
#define se second
#define pi acos(-(long double)1.0)
#define LL long long
//#define mp make_pair
#define pb push_back
#define ls rt<<1, l, m
#define rs rt<<1|1, m+1, r
#define ULL unsigned LL
#define pll pair<LL, LL>
#define pli pair<LL, int>
#define pii pair<int, int>
#define piii pair<pii, int>
#define pdd pair<long double, long double>
#define mem(a, b) memset(a, b, sizeof(a))
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define fopen freopen("in.txt", "r", stdin);freopen("out.txt", "w", stout);
//head char s[], mp[][];
int cnt = , ansx, ansy;
int check(char c) {
for (int i = ; i < ; i++) {
int a = ;
for (int j = ; j < ; j++) {
if(mp[i][j] == c) a++;
}
if(a == ) return ;
a = ;
for (int j = ; j < ; j++) {
if(mp[j][i] == c) a++;
}
if(a == ) return ;
}
int a = ;
for (int i = ; i < ; i++) if(mp[i][i] == c) a++;
if(a == ) return ;
a = ;
for (int i = ; i < ; i++) if(mp[i][-i] == c) a++;
if(a == ) return ;
return ;
}
int dfs(int step, int a, int b) {
if((cnt-step)% == ) {
int tmp = check('o');
if(tmp || step == ) return -tmp;
}
else {
int tmp = check('x');
if(tmp || step == ) return tmp;
}
if((cnt-step)% == ) {
for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
if(mp[i][j] == '.') {
mp[i][j] = 'x';
int tmp = dfs(step-, a, b);
mp[i][j] = '.';
if(tmp >= a) {
if(step == cnt) {
ansx = i;
ansy = j;
}
a = tmp;
if(b <= a) return b;
}
}
}
}
return a;
}
else {
for (int i = ; i < ; i++) {
for (int j = ; j < ; j++) {
if(mp[i][j] == '.') {
mp[i][j] = 'o';
int tmp = dfs(step-, a, b);
mp[i][j] = '.';
if(tmp <= b) {
b = tmp;
if(a >= b) return a;
}
}
}
}
return b;
}
}
int main() {
while(~scanf("%s", s)) {
if(s[] == '$') return ;
for (int i = ; i < ; i++) scanf("%s", mp[i]);
cnt = ;
for (int i = ; i < ; i++) for (int j = ; j < ; j++) if(mp[i][j] == '.') cnt++;
if(cnt >= ) {
printf("#####\n");
continue;
}
int ans = dfs(cnt, -, );
if(ans == ) printf("(%d,%d)\n", ansx, ansy);
else printf("#####\n");
}
return ;
}
例题2:POJ - 1085
思路1:记忆化搜索dp, dp[s]表示从状态s出发先手所能获得的最大利益。
特点:空间大,时间短
代码:
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize(4)
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<climits>
using namespace std;
#define fi first
#define se second
#define pi acos(-1.0)
#define LL long long
//#define mp make_pair
#define pb push_back
#define ls rt<<1, l, m
#define rs rt<<1|1, m+1, r
#define ULL unsigned LL
#define pll pair<LL, LL>
#define pli pair<LL, int>
#define pii pair<int, int>
#define piii pair<pii, int>
#define pdd pair<long double, long double>
#define mem(a, b) memset(a, b, sizeof(a))
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define fopen freopen("in.txt", "r", stdin);freopen("out.txt", "w", stout);
//head int line[][] = {{, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }};
int tri[][] = {{, , }, {, , }, {, , }, {, , }, {, , }, {, , }, {, , }, {, , }, {, , }};
int num[][];
int dp[<<];
int count(int s) {
int cnt = ;
for (int i = ; i < ; i++) if((s&(<<tri[i][])) && (s&(<<tri[i][])) && (s&(<<tri[i][]))) cnt++;
return cnt;
}
int dfs(int s) {
if(~dp[s]) return dp[s];
if(__builtin_popcount(s) == ) return ;
int prenum = count(s);
int ans = INT_MIN;
for(int i = ; i < ; i++) {
if(!(s&(<<i))) {
int nownum = count(s|(<<i));
if(nownum == prenum) {
ans = max(ans, -dfs(s|<<i));
}
else {
ans = max(ans, nownum - prenum + dfs(s|<<i));
}
}
}
return dp[s] = ans;
}
int main() {
int T, m, u, v;
for (int i = ; i < ; i++) num[line[i][]][line[i][]] = num[line[i][]][line[i][]] = i;
mem(dp, -);
scanf("%d", &T);
for(int cs = ; cs <= T; cs++) {
int st = ;
scanf("%d", &m);
int prenum = , nownum = ;
int cnt = , step = ;
for (int i = ; i <= m; i++) {
scanf("%d %d", &u, &v);
st |= <<num[u][v];
nownum = count(st);
if(nownum > prenum) {
if(step% == ) cnt += nownum - prenum;
else cnt -= nownum - prenum;
}
else step++;
prenum = nownum;
}
if(step% == ) cnt += dfs(st);
else cnt -= dfs(st);
if(cnt > ) printf("Game %d: A wins.\n", cs);
else printf("Game %d: B wins.\n", cs);
}
return ;
}
思路2:极大极小搜索+alpha-beta剪枝
特点:空间小,时间长
代码:
#pragma GCC optimize(2)
#pragma GCC optimize(3)
#pragma GCC optimize(4)
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<climits>
using namespace std;
#define fi first
#define se second
#define pi acos(-1.0)
#define LL long long
//#define mp make_pair
#define pb push_back
#define ls rt<<1, l, m
#define rs rt<<1|1, m+1, r
#define ULL unsigned LL
#define pll pair<LL, LL>
#define pli pair<LL, int>
#define pii pair<int, int>
#define piii pair<pii, int>
#define pdd pair<long double, long double>
#define mem(a, b) memset(a, b, sizeof(a))
#define fio ios::sync_with_stdio(false);cin.tie(0);cout.tie(0);
#define fopen freopen("in.txt", "r", stdin);freopen("out.txt", "w", stout);
//head int line[][] = {{, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }, {, }};
int tri[][] = {{, , }, {, , }, {, , }, {, , }, {, , }, {, , }, {, , }, {, , }, {, , }};
int num[][];
int count(int s) {
int cnt = ;
for (int i = ; i < ; i++) if((s&(<<tri[i][])) && (s&(<<tri[i][])) && (s&(<<tri[i][]))) cnt++;
return cnt;
}
int dfs(int s, int a, int b, int alpha, int beta, int step) {
if(__builtin_popcount(s) == ) {
if(a > b) return ;
else return -;
}
int prenum = count(s);
for(int i = ; i < ; i++) {
if(!(s&(<<i))) {
int nownum = count(s|(<<i));
if(nownum == prenum) {
if(step% == ) {
int tmp = dfs(s|<<i, a, b, alpha, beta, step+);
if(tmp >= alpha) {
alpha = tmp;
if(alpha >= beta) return alpha;
}
}
else {
int tmp = dfs(s|<<i, a, b, alpha, beta, step+);
if(tmp <= beta) {
beta = tmp;
if(alpha >= beta) return beta;
}
}
}
else {
if(step% == ) {
int tmp = dfs(s|<<i, a+nownum-prenum, b, alpha, beta, step);
if(tmp >= alpha) {
alpha = tmp;
if(alpha >= beta) return alpha;
}
}
else {
int tmp = dfs(s|<<i, a, b+nownum-prenum, alpha, beta, step);
if(tmp <= beta) {
beta = tmp;
if(alpha >= beta) return beta;
}
}
}
}
}
if(step% == ) return alpha;
else return beta;
}
int main() {
int T, m, u, v;
for (int i = ; i < ; i++) num[line[i][]][line[i][]] = num[line[i][]][line[i][]] = i;
scanf("%d", &T);
for(int cs = ; cs <= T; cs++) {
int st = ;
scanf("%d", &m);
int prenum = , nownum = ;
int step = , a = , b = ;
for (int i = ; i <= m; i++) {
scanf("%d %d", &u, &v);
st |= <<num[u][v];
nownum = count(st);
if(nownum > prenum) {
if(step% == ) a += nownum - prenum;
else b += nownum - prenum;
}
else step++;
prenum = nownum;
}
int cnt = dfs(st, a, b, -, , step);
if(cnt > ) printf("Game %d: A wins.\n", cs);
else printf("Game %d: B wins.\n", cs);
}
return ;
}
算法笔记--极大极小搜索及alpha-beta剪枝的更多相关文章
- POJ 1568 极大极小搜索 + alpha-beta剪枝
极小极大搜索 的个人理解(alpha-beta剪枝) 主要算法依据就是根据极大极小搜索实现的. 苦逼的是,查了两个晚上的错,原来最终是判断函数写错了..瞬间吐血! ps. 据说加一句 if sum & ...
- 极大极小搜索思想+(α/β)减枝 【转自-----https://blog.csdn.net/hzk_cpp/article/details/79275772】
极大极小搜索,即minimax搜索算法,专门用来做博弈论的问题的暴力. 多被称为对抗搜索算法. 这个搜索算法的基本思想就是分两层,一层是先手,记为a,还有一层是后手,记为b. 这个搜索是认为这a与b的 ...
- 软件发布版本区别介绍-Alpha,Beta,RC,Release
Alpha: Alpha是内部测试版,一般不向外部发布,会有很多Bug.除非你也是测试人员,否则不建议使用. 是希腊字母的第一位,表示最初级的版本 alpha就是α,beta就是β alpha版就是比 ...
- 算法笔记--数位dp
算法笔记 这个博客写的不错:http://blog.csdn.net/wust_zzwh/article/details/52100392 数位dp的精髓是不同情况下sta变量的设置. 模板: ]; ...
- 算法笔记之KMP算法
本文是<算法笔记>KMP算法章节的阅读笔记,文中主要内容来源于<算法笔记>.本文主要介绍了next数组.KMP算法及其应用以及对KMP算法的优化. KMP算法主要用于解决字符串 ...
- [转载]SharePoint 2013搜索学习笔记之搜索构架简单概述
Sharepoint搜索引擎主要由6种组件构成,他们分别是爬网组件,内容处理组件,分析处理组件,索引组件,查询处理组件,搜索管理组件.可以将这6种组件分别部署到Sharepoint场内的多个服务器上, ...
- poj 1568 Find the Winning Move 极大极小搜索
思路:用极大极小搜索解决这样的问题很方便!! 代码如下: #include <cstdio> #include <algorithm> #define inf 10000000 ...
- 学习Java 以及对几大基本排序算法(对算法笔记书的研究)的一些学习总结(Java对算法的实现持续更新中)
Java排序一,冒泡排序! 刚刚开始学习Java,但是比较有兴趣研究算法.最近看了一本算法笔记,刚开始只是打算随便看看,但是发现这本书非常不错,尤其是对排序算法,以及哈希函数的一些解释,让我非常的感兴 ...
- 软工+C(2017第4期) Alpha/Beta换人
// 上一篇:超链接 // 下一篇:工具和结构化 注:在一次软件工程讨论课程进度设计的过程中,出现了这个关于 Alpha/Beta换人机制的讨论,这个机制在不同学校有不同的实施,本篇积累各方观点,持续 ...
随机推荐
- 最短路径spfa
#include <stdio.h> #include <string.h> #include <algorithm> #include <queue> ...
- HTTP协议实际使用笔记
mozilla的帮助文档: https://developer.mozilla.org/zh-CN/docs/Web/HTTP HTTP协议详解(转) php http头设置相关信息 这个2篇最好先看 ...
- AngularJS实现可伸缩的页面切换
AngularJS实现可伸缩的页面切换 AngularJS 1.2 通过引入基于纯CSS class的切换和动画,在一个单页面应用创建页面到页面的切换变得更加的容易.只需要使用一个ng-view,让我 ...
- angular--获取时间方法services
写了一些公用方法获取自然周.传入开始和结束日期,获取中间全部日期等方法 .service('DateServices', [function () { // 获取某年自然周的方法 (如果是当年,只返回 ...
- java EE(1)
1.安装eclipse javaEE版本,配置好Tomcat服务器(略) 2.新建项目: Dynamic web project 3.创建服务器,并选择项目发布目录 4.添加项目: 右键服务器,选择A ...
- 16.1-uC/OS-III同步 (任务内建信号量)
1.经常通过发送信号量实现同步.每个任务都有内建的信号量,通过任务内建的信号量不仅可以简化信号量通信的代码而且更加有效. 与任务内建的信号量相关的函数都是以 OSTaskSem???()为前缀的.相关 ...
- 【Tools】-NO.93.Tools.5.pyCharm-【pyCharm 安装与破解配置】-
1.0.0 Summary Tittle:[Tools]-NO.93.Tools.5.pyCharm-[pyCharm 安装与破解配置]- Style:Tools Series:Tools Since ...
- iDigital2019数字营销广告主峰会
iDigital 数字营销品牌广告主峰会是定向邀请形式的闭门社交峰会, 现已成长为享誉亚洲的营销广告行业活动, 致力于帮助品牌营销官, 决策者和业界同行在日益整合的数字商业时代下获得战略远景.组委会邀 ...
- C#:进程
using System; using System.Collections.Generic; using System.Diagnostics; using System.Linq; using S ...
- WAR包方式安装Jenkins
WAR包方式安装Jenkins 系统环境: CentOS 7.5 1804 IP:192.168.1.3 关闭selinux.firewalld jenkins war包:下载地址 一.安装t ...