首页 > 其他 > 详细

Luogu T103187 东方CannonBall 概率dp

时间:2019-10-13 15:40:35      阅读:115      评论:0      收藏:0      [点我收藏+]

题目背景

东方CannonBall(译名东方大炮弹)是近期发布的第一款获得ZUN授权的商业东方二次创作手机游戏,游戏关卡中玩家在类似大富翁的棋盘上与CPU或是玩家进行对战。

题目描述

在游戏中,每位玩家拥有若干个六面骰子。当两名玩家相遇时,将会通过掷出所有骰子的方式判定胜负。你拥有xx个骰子,而你的对手拥有yy个骰子,请问你胜率,即你的点数和大于对手的概率是多少?

正式地说,将玩家的nn个骰子编号为1..n1..n,在每次掷骰子时,第ii个骰子的点数a_iai?会在\{1,2,3,4,5,6\}{1,2,3,4,5,6}中等概率随机选取,玩家的点数和C_n=\sum_{1\le i \le n} a_iCn?=1in?ai?,求P(C_x>C_y)P(Cx?>Cy?)

输入格式

一行2个整数,依次表示你的骰子数xx和对手的骰子数yy。

输出格式

一行一个百分数,四舍五入保留小数点后2位小数,表示你的胜率。

输入输出样例

输入 #1
1 1
输出 #1
41.67%
输入 #2
500 499
输出 #2
52.22%

说明/提示

对于样例1,双方有\frac {1} {6}61?的概率掷出相同值,其他情况下输赢各占一半,故答案为\frac {5} {12}125?

本题使用文本方式比较

数据规模

对于 30\%30% 的数据,保证 x+y \le 8x+y8。

对于 60\%60% 的数据,保证 x,y \le 10x,y10

对于 100\%100% 的数据,保证 1 \le x, y\leq 10001x,y1000

做法如题。100分目测需要加高精度(+ × / 运算)

技术分享图片
 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cstring>
 4 #include<algorithm>
 5 
 6 using namespace std;
 7 typedef long long ll;
 8 
 9 ll read(){
10     ll a = 0; char l =  ,c = getchar();
11     while(c < 0||c > 9)l = c,c = getchar();
12     while(0 <= c&&c <= 9)a = a*10+c-0,c = getchar();
13     if(l == -)return -a; return a;
14 }
15 
16 ll f[2][2][6010],s[2][6010];
17 ll x,y,win,tot;
18 
19 int main(){
20     x = read(),y = read();
21 //    for(int i = 1;i <= 6;i++)f[0][0][i] = f[0][1][i] = 1;
22     f[0][0][0] = f[0][1][0] = 1;
23     for(int i = 1;i <= x;i++){
24         memset(f[i&1][0],0,sizeof(f[i&1][0]));
25         for(int j = x*6;j >= 0;j--)if(f[~i&1][0][j])
26             for(int k = 1;k <= 6;k++)
27                 f[i&1][0][j+k] += f[~i&1][0][j];
28     }
29     for(int i = 1;i <= y;i++){
30         memset(f[i&1][1],0,sizeof(f[i&1][1]));
31         for(int j = y*6;j >= 0;j--)if(f[~i&1][1][j]){
32 //            printf("%d ",f[~i&1][1][j]);
33             for(int k = 1;k <= 6;k++)
34                 f[i&1][1][j+k] += f[~i&1][1][j];
35         }//putchar(‘\n‘);
36     }
37 //    for(int i = 1;i <= x*6;i++)
38 //        s[0][i] = s[0][i-1]+f[0][i];
39 //    for(int i = 1;i <= y*6;i++)
40 //        s[1][i] = s[1][i-1]+f[y&1][1][i];
41     for(int i = x;i <= x*6;i++){
42         ll cur = 0;
43         for(int j = y;j < i;j++)cur += f[x&1][0][i]*f[y&1][1][j];
44         win += cur;
45         for(int j = i;j <= y*6;j++)cur += f[x&1][0][i]*f[y&1][1][j];
46         tot += cur;
47     }
48     
49 //    cout << win << ‘ ‘ << tot << endl;
50 //    for(int i = 0;i <= x*6;i++)printf("%d ",f[x&1][0][i]);putchar(‘\n‘);
51 //    for(int i = 0;i <= y*6;i++)printf("%d ",f[y&1][1][i]);putchar(‘\n‘);
52     
53     double ans = 1.0*win/tot*100;
54     printf("%.2f%%",ans);
55 return 0;
56 }
60分代码

 

Luogu T103187 东方CannonBall 概率dp

原文:https://www.cnblogs.com/Wangsheng5/p/11666536.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!