题意:三国杀,给定4个白板武将的血量,4个角色轮流行动,每回合行动时如果该人存活则可以选择使阵营不同的角色血量-1,血量为0则死亡。每个人按自己获胜概率最大化行动,如果有多种方案概率相同则等概率选择这些策略,问最后主公、反贼、内奸三个阵营分别获胜的概率

0 < h1 < 40, 0 ≤ h2 < 40, 0 ≤ h3 < 40, 0 ≤ h4 < 40

思路:当前的决策与概率只与血量和当前行动的人有关

注意行动顺序是主公,反贼,忠臣,内奸

其他的具体看代码注释吧

 #include<cstdio>
#include<cstring>
#include<string>
#include<cmath>
#include<iostream>
#include<algorithm>
#include<map>
#include<set>
#include<queue>
#include<vector>
using namespace std;
typedef long long ll;
typedef unsigned int uint;
typedef unsigned long long ull;
typedef pair<int,int> PII;
typedef vector<int> VI;
#define fi first
#define se second
#define MP make_pair
#define N 11000
#define M 41
#define MOD 1e9+7
#define eps 1e-13
#define pi acos(-1) double dp[][][][][][];
bool vis[][][][][]; int read()
{
int v=,f=;
char c=getchar();
while(c<||<c) {if(c=='-') f=-; c=getchar();}
while(<=c&&c<=) v=(v<<)+v+v+c-,c=getchar();
return v*f;
} void dfs(int a,int b,int c,int d,int i)
{
if(vis[a][b][c][d][i]) return; vis[a][b][c][d][i]=; if(!c&&!d) //Both of the Rebel and the Traitor are dead
{
dp[a][b][c][d][i][]=1.0;
dp[a][b][c][d][i][]=dp[a][b][c][d][i][]=0.0;
return; //Monarch and the Minister win
} if(!a) //The Monarch is dead
{
if(a+b+c==) //If the Traitor is surviving and all the other players are dead
{
dp[a][b][c][d][i][]=1.0;
dp[a][b][c][d][i][]=dp[a][b][c][d][i][]=0.0;
return; //the Traitor wins
}
else
{
dp[a][b][c][d][i][]=1.0;
dp[a][b][c][d][i][]=dp[a][b][c][d][i][]=0.0;
return; //the Rebel wins
}
} if(i==&&a==||i==&&c==||i==&&b==||i==&&d==) //deadman turn
{
dfs(a,b,c,d,(i+)%);
for(int j=;j<=;j++) dp[a][b][c][d][i][j]=dp[a][b][c][d][(i+)%][j];
return;
} int j=(i+)%;
double s1=0.0;
double s2=0.0;
double s3=0.0;
int x=; if(i==||i==) //Monarch and the Minister
{
double t=-1.0; if(c)
{
dfs(a,b,c-,d,j);
double k=dp[a][b][c-][d][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a][b][c-][d][j][];
s2=dp[a][b][c-][d][j][];
s3=dp[a][b][c-][d][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a][b][c-][d][j][];
s2+=dp[a][b][c-][d][j][];
s3+=dp[a][b][c-][d][j][];
}
} if(d)
{
dfs(a,b,c,d-,j);
double k=dp[a][b][c][d-][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a][b][c][d-][j][];
s2=dp[a][b][c][d-][j][];
s3=dp[a][b][c][d-][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a][b][c][d-][j][];
s2+=dp[a][b][c][d-][j][];
s3+=dp[a][b][c][d-][j][];
}
}
} if(i==) //Rebel
{
double t=-1.0; if(a)
{
dfs(a-,b,c,d,j);
double k=dp[a-][b][c][d][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a-][b][c][d][j][];
s2=dp[a-][b][c][d][j][];
s3=dp[a-][b][c][d][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a-][b][c][d][j][];
s2+=dp[a-][b][c][d][j][];
s3+=dp[a-][b][c][d][j][];
}
} if(b)
{
dfs(a,b-,c,d,j);
double k=dp[a][b-][c][d][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a][b-][c][d][j][];
s2=dp[a][b-][c][d][j][];
s3=dp[a][b-][c][d][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a][b-][c][d][j][];
s2+=dp[a][b-][c][d][j][];
s3+=dp[a][b-][c][d][j][];
}
} if(d)
{
dfs(a,b,c,d-,j);
double k=dp[a][b][c][d-][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a][b][c][d-][j][];
s2=dp[a][b][c][d-][j][];
s3=dp[a][b][c][d-][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a][b][c][d-][j][];
s2+=dp[a][b][c][d-][j][];
s3+=dp[a][b][c][d-][j][];
}
}
} if(i==) //Traitor
{
double t=-1.0; if(a)
{
dfs(a-,b,c,d,j);
double k=dp[a-][b][c][d][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a-][b][c][d][j][];
s2=dp[a-][b][c][d][j][];
s3=dp[a-][b][c][d][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a-][b][c][d][j][];
s2+=dp[a-][b][c][d][j][];
s3+=dp[a-][b][c][d][j][];
}
} if(b)
{
dfs(a,b-,c,d,j);
double k=dp[a][b-][c][d][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a][b-][c][d][j][];
s2=dp[a][b-][c][d][j][];
s3=dp[a][b-][c][d][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a][b-][c][d][j][];
s2+=dp[a][b-][c][d][j][];
s3+=dp[a][b-][c][d][j][];
}
} if(c)
{
dfs(a,b,c-,d,j);
double k=dp[a][b][c-][d][j][]; if(k-t>eps)
{
x=;
t=k;
s1=dp[a][b][c-][d][j][];
s2=dp[a][b][c-][d][j][];
s3=dp[a][b][c-][d][j][];
}
else
if(fabs(k-t)<eps)
{
x++;
s1+=dp[a][b][c-][d][j][];
s2+=dp[a][b][c-][d][j][];
s3+=dp[a][b][c-][d][j][];
}
}
} dp[a][b][c][d][i][]=s1/x;
dp[a][b][c][d][i][]=s2/x;
dp[a][b][c][d][i][]=s3/x;
//printf("%d %d %d %d %d %.3lf %.3lf %.3lf\n",a,b,c,d,i,dp[a][b][c][d][i][0],dp[a][b][c][d][i][1],dp[a][b][c][d][i][2]);
} int main()
{
freopen("hdoj6224.in","r",stdin);
freopen("hdoj6224.out","w",stdout);
int cas;
scanf("%d",&cas); memset(vis,,sizeof(vis)); while(cas--)
{
int A,B,C,D;
scanf("%d%d%d%d",&A,&B,&C,&D);
dfs(A,B,C,D,);
printf("%.6lf %.6lf %.6lf\n",dp[A][B][C][D][][],dp[A][B][C][D][][],dp[A][B][C][D][][]);
}
return ;
}

【HDOJ6224】Legends of the Three Kingdoms(概率DP)的更多相关文章

  1. Codeforces 28C [概率DP]

    /* 大连热身D题 题意: 有n个人,m个浴室每个浴室有ai个喷头,每个人等概率得选择一个浴室. 每个浴室的人都在喷头前边排队,而且每个浴室内保证大家都尽可能均匀得在喷头后边排队. 求所有浴室中最长队 ...

  2. HDU 4405 Aeroplane chess (概率DP)

    题意:你从0开始,要跳到 n 这个位置,如果当前位置是一个飞行点,那么可以跳过去,要不然就只能掷骰子,问你要掷的次数数学期望,到达或者超过n. 析:概率DP,dp[i] 表示从 i  这个位置到达 n ...

  3. POJ 2096 Collecting Bugs (概率DP)

    题意:给定 n 类bug,和 s 个子系统,每天可以找出一个bug,求找出 n 类型的bug,并且 s 个都至少有一个的期望是多少. 析:应该是一个很简单的概率DP,dp[i][j] 表示已经从 j ...

  4. POJ 2151 Check the difficulty of problems (概率DP)

    题意:ACM比赛中,共M道题,T个队,pij表示第i队解出第j题的概率 ,求每队至少解出一题且冠军队至少解出N道题的概率. 析:概率DP,dp[i][j][k] 表示第 i 个队伍,前 j 个题,解出 ...

  5. 概率DP light oj 1030

    t组数据 n块黄金 到这里就捡起来 出发点1 到n结束  点+位置>n 重掷一次 dp[i] 代表到这里的概率 dp[i]=(dp[i-1]+dp[i-2]... )/6  如果满6个的话 否则 ...

  6. hdu 4050 2011北京赛区网络赛K 概率dp ***

    题目:给出1-n连续的方格,从0开始,每一个格子有4个状态,左右脚交替,向右跳,而且每一步的步长必须在给定的区间之内.当跳出n个格子或者没有格子可以跳的时候就结束了,求出游戏的期望步数 0:表示不能到 ...

  7. [转]概率DP总结 by kuangbin

    概率类题目一直比较弱,准备把kuangbin大师傅总结的这篇题刷一下! 我把下面的代码换成了自己的代码! 原文地址:http://www.cnblogs.com/kuangbin/archive/20 ...

  8. SGU 422 Fast Typing(概率DP)

    题目大意 某人在打字机上打一个字符串,给出了他打每个字符出错的概率 q[i]. 打一个字符需要单位1的时间,删除一个字符也需要单位1的时间.在任意时刻,他可以花 t 的时间检查整个打出来的字符串,并且 ...

  9. HDU 4050 wolf5x(动态规划-概率DP)

    wolf5x Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Sub ...

  10. 概率dp入门

    概率DP主要用于求解期望.概率等题目. 转移方程有时候比较灵活. 一般求概率是正推,求期望是逆推.通过题目可以体会到这点. poj2096:Collecting Bugs #include <i ...

随机推荐

  1. 【状态压缩dp】1195: [HNOI2006]最短母串

    一个清晰的思路就是状压dp:不过也有AC自动机+BFS的做法 Description 给定n个字符串(S1,S2,„,Sn),要求找到一个最短的字符串T,使得这n个字符串(S1,S2,„,Sn)都是T ...

  2. python入门:输出1-100之内的所有奇数和偶数

    #!/usr/bin/env python # -*- coding:utf-8 -*- #输出1-100之内的所有奇数和偶数 """ 给start赋值等于1,while ...

  3. 拓展jQuery的serialize(),将form表单转化为json对象

    jQuery 的 serialize() 方法经常会报 Uncaught TypeError: JSON.serializeObject is not a function 的错误, 原装的方法真的一 ...

  4. 局域网映射到公网-natapp实现

    在开发时可能会有这样的需求: 需要将自己开发的机器上的应用提供到公网上进行访问,但是并不想通过注册域名.搭建服务器等等一系列繁琐的操作来实现. 例如:微信公众号的开发调试就需要用到域名访问本机项目. ...

  5. 22.Yii2.0框架多表关联一对一查询之hasOne

    思路: 通过文章查它对应的分类信息 一对一的关系 控制器里 //一对一关联查询 public function actionRelatesone() { //方法一,hasOne() 用查一条文章的结 ...

  6. GoF23种设计模式之行为型模式之中介者模式

    一.概述 使用一个中介对象来封装一系列的对象交互.中介者让各个对象无需显式地相互引用,从而达到解耦的效果.并且可以独立地改变它们之间的交互.二.适用性1.当一组对象以定义良好但复杂通信的时候.产生的相 ...

  7. ACM Changchun 2015 J. Chip Factory

    John is a manager of a CPU chip factory, the factory produces lots of chips everyday. To manage larg ...

  8. re--findall 【转】

    原文链接 python re 模块 findall 函数用法简述 代码示例: >>> import re >>> s = "adfad asdfasdf ...

  9. Linux学习-什么是登录档

    CentOS 7 登录档简易说明 登录档的重要性 为什么说登录文件很重要, 解决系统方面的错误: 用 Linux 这么久了,你应该偶而会发现系统可能会出现一些错误,包括硬件捉不到或者是某些系 统服务无 ...

  10. Monkeyrunner脚本的录制与回放

    继上一篇monkeyrunner环境搭建:http://www.cnblogs.com/zh-ya-jing/p/4351245.html 之后,我们可以进一步学习monkeyrunner了. 我也是 ...