论文参考 汤可因《浅谈一类数学期望问题的解决方法》

反正是很神奇的东西吧。。我脑子不好不是很能想得到。

bzoj 1415

1415: [Noi2005]聪聪和可可

Time Limit: 10 Sec  Memory Limit: 162 MB
Submit: 2140  Solved: 1258
[Submit][Status][Discuss]

Description

Input

数据的第1行为两个整数N和E,以空格分隔,分别表示森林中的景点数和连接相邻景点的路的条数。 第2行包含两个整数C和M,以空格分隔,分别表示初始时聪聪和可可所在的景点的编号。 接下来E行,每行两个整数,第i+2行的两个整数Ai和Bi表示景点Ai和景点Bi之间有一条路。 所有的路都是无向的,即:如果能从A走到B,就可以从B走到A。 输入保证任何两个景点之间不会有多于一条路直接相连,且聪聪和可可之间必有路直接或间接的相连。

Output

输出1个实数,四舍五入保留三位小数,表示平均多少个时间单位后聪聪会把可可吃掉。

Sample Input

【输入样例1】
4 3
1 4
1 2
2 3
3 4
【输入样例2】
9 9
9 3
1 2
2 3
3 4
4 5
3 6
4 6
4 7
7 8
8 9

Sample Output

【输出样例1】
1.500
【输出样例2】
2.167
 

 
论文题。。自己看看论文就懂了。
 
 
 
 
 #include<bits/stdc++.h>
#define clr(x) memset(x,0,sizeof(x))
#define clr_1(x) memset(x,-1,sizeof(x))
#define INF 0x3f3f3f3f
#define mod 1000000009
#define LL long long
#define next nexted
using namespace std;
const int N=1e3+;
int ans[N][N],vis[N],step[N][N];
double f[N][N];
int u,v,m,n,pu,pv;
vector<int> edge[N];
void bfs(int root)
{
int now,p;
clr(vis);
queue<int> que;
step[root][root]=root;
vis[root]=;
for(int i=;i<edge[root].size();i++)
{
p=edge[root][i];
que.push(p);
step[root][p]=p;
vis[p]=;
}
while(!que.empty())
{
now=que.front();
que.pop();
for(int i=;i<edge[now].size();i++)
{
p=edge[now][i];
if(!vis[p])
{
que.push(p);
vis[p]=;
step[root][p]=step[root][now];
}
}
}
return ;
}
double Find(int u,int v)
{
if(u==v)
return f[u][v]=;
if(f[u][v]>)
return f[u][v];
if(step[u][v]==v)
return f[u][v]=;
if(step[step[u][v]][v]==v)
return f[u][v]=;
int p;
for(int i=;i<edge[v].size();i++)
{
p=edge[v][i];
f[u][v]+=Find(step[step[u][v]][v],p);
}
f[u][v]+=Find(step[step[u][v]][v],v);
f[u][v]=f[u][v]/(edge[v].size()+)+;
return f[u][v];
}
int main()
{
scanf("%d%d",&n,&m);
scanf("%d%d",&u,&v);
for(int i=;i<=m;i++)
{
scanf("%d%d",&pu,&pv);
edge[pu].push_back(pv);
edge[pv].push_back(pu);
}
for(int i=;i<=n;i++)
sort(edge[i].begin(),edge[i].end());
for(int i=;i<=n;i++)
bfs(i);
printf("%.3f\n",Find(u,v));
return ;
}

bzoj 2685

2685: Sgu385 highlander

Time Limit: 10 Sec  Memory Limit: 128 MBSec  Special Judge
Submit: 112  Solved: 64
[Submit][Status][Discuss]

Description

一个游戏N个人,每个人开始一张卡片,上面写着N个人中某个人的名字。
每张卡片上的名字都不同,且不会拿到自己名字的卡片。
游戏开始时,每个人开始追自己卡片上写着的人,如果A有写着B的卡片。
当A追到B后,A可以拿到所有B的卡片。如果每个人都没人可追,游戏结束。
这时开始数每个人手上的卡片总数,获得卡片最多的人即是胜者。如果有多个人
的卡片一样多,则都是胜者。现想知道有多少人在理论上有概论成为胜者。

Input

输入一个整数N 2<=N<=100

Output

一个实数

Sample Input

2

Sample Output

2

HINT

你的答案与标准答案的差不超过10^-9

Source


同样的一道论文题。

这个题目我们可以把这些错排看成一张有向图。由于每个点出入度为1,且不会指向自己。所以这是一个多个不相交的环,并且不存在自环(这个很重要,公式推导的时候环长度>1)组成的图。

那我们看看数据范围n≤100,所以能接受O(n3)的算法。

于是我们做这么一个三维的推导:

f[i][j][k]代表有i个点确定,其中最长的环长度为j,且长度为j的环数量为k的情况数量,K也为获胜人数。

如果我们只形成一个长度为j的环,他的情况数量为A(n,j)/j,n为剩余点的数量。

如果我们k个长度为j的环,他的情况数量为(A(n,j)*A(n-j,j)*A(n-2*j,j)……A(n-(k-1)*j,j))/A(j,j),n为剩余点数量。因为A(j,j)==j!==1*2*3……*j,所以我们从k=1推到k=n的时候f[i][j][k]=f[i-j][j][k-1]*A(n-(i-j),j)/j。

然后g[i][j]代表i个点确定,最长环长度为j的情况数量,即把k那一维求和。

可得出以下公式:

答案分母为情况总数,即错排总数,可以通过错排递推式得到:

他是所有和项的分母。他和上面的j*f[i][j][k]形成每种获胜人数下的权(概率)。

然后j*k是每种情况的下获胜人数,就是环里所有人都可能赢,没毛病。

相当于运用全概率公式E(K)=E(E(K|J)=|J=1)+E(K|J=2)……E(K|J=n)

 #include<bits/stdc++.h>
#define clr(x) memset(x,0,sizeof(x))
#define clr_1(x) memset(x,-1,sizeof(x))
#define mod 1000000007
#define LL long long
#define INF 0x3f3f3f3f
using namespace std;
const int N=1e2+;
double f[N][N][N],g[N][N];
double pre[],ans;
double jc[N];
int n,m,T,p;
double a(int n,int m)
{
return n>=m?jc[n]/jc[n-m]:;
}
int main()
{
scanf("%d",&n);
pre[]=pre[]=;
pre[]=;
for(int i=;i<=n;i++)
pre[i%]=(i-)*(pre[(i-)%]+pre[(i-)%]);
jc[]=jc[]=;
for(int i=;i<=n;i++)
jc[i]=jc[i-]*i;
for(int i=;i<=n;i++)
{
for(int j=;j<i;j++)
{
p=min(i-j,j-);
for(int k=;k<=p;k++)
f[i][j][]+=g[i-j][k];
f[i][j][]*=a(n-i+j,j)/j;
g[i][j]=f[i][j][];
p=i/j;
for(int k=;k<=p;k++)
g[i][j]+=(f[i][j][k]=f[i-j][j][k-]*a(n-i+j,j)/j/k);
}
f[i][i][]=g[i][i]=a(n,i)/i;
}
for(int i=;i<=n;i++)
{
p=n/i;
for(int j=;j<=p;j++)
ans+=f[n][i][j]*j*i;
}
ans/=pre[n%];
printf("%.10f\n",ans);
}

bzoj 1419

1419: Red is good

Time Limit: 10 Sec  Memory Limit: 64 MB
Submit: 1210  Solved: 560
[Submit][Status][Discuss]

Description

桌面上有R张红牌和B张黑牌,随机打乱顺序后放在桌面上,开始一张一张地翻牌,翻到红牌得到1美元,黑牌则付
出1美元。可以随时停止翻牌,在最优策略下平均能得到多少钱。

Input

一行输入两个数R,B,其值在0到5000之间

Output

在最优策略下平均能得到多少钱。

Sample Input

5 1

Sample Output

4.166666

HINT

输出答案时,小数点后第六位后的全部去掉,不要四舍五入.

Source

 

 
这个人的决策只有两种:翻开牌和不翻开牌。所谓的最优策略。。还有什么最优?当然是期望得到的钱最优啊。因此在I张R和J张B的情况下,如果我们翻牌的期望>0也就是还能拿钱的情况下我们选择继续翻牌来获得金钱。反之不翻牌,期望为0。
设f[i,j]表示还剩I张R牌和J张B牌的期望,边界条件是f[i,0]=f[i-1,0]+1,f[0,j]=0。转移方程是$f[i][j]=(f[i-1][j]+1)*\frac{i}{i+j}+(f[i][j-1]-1)*\frac{j}{i+j}$。
 #include<bits/stdc++.h>
#define clr(x) memset(x,0,sizeof(x))
#define clr_1(x) memset(x,-1,sizeof(x))
#define mod 1000000007
#define LL long long
#define INF 0x3f3f3f3f
using namespace std;
const int N=5e3+;
int n,m;
double f[][N];
int main()
{
scanf("%d%d",&n,&m);
for(int i=;i<=n;i++)
{
f[i&][]=f[i&^][]+;
for(int j=;j<=m;j++)
f[i&][j]=max(0.0,((f[i&][j-]-)*j+(f[i&^][j]+)*i)/(i+j));
}
printf("%.6f\n",((LL)(f[n&][m]*))*1.0/);
}

概率dp学习记录的更多相关文章

  1. DP学习记录Ⅰ

    DP学习记录Ⅱ 前言 状态定义,转移方程,边界处理,这三部分想好了,就问题不大了.重点在状态定义,转移方程是基于状态定义的,边界处理是方便转移方程的开始的.因此最好先在纸上写出自己状态的意义,越详细越 ...

  2. DP学习记录Ⅱ

    DP学习记录Ⅰ 以下为 DP 的优化. 人脑优化DP P5664 Emiya 家今天的饭 正难则反.考虑计算不合法方案.一个方案不合法一定存在一个主食,使得该主食在多于一半的方法中出现. 枚举这个&q ...

  3. 概率dp学习

    预备知识 一.期望的数学定义 如果X 是一个离散的随机变量,输出值为 x1, x2, ..., 和输出值相应的概率为p1, p2, ... (概率和为 1), 那么期望值为E(x)=x1p1+x2p2 ...

  4. HDU 4405:Aeroplane chess(概率DP入门)

    http://acm.split.hdu.edu.cn/showproblem.php?pid=4405 Aeroplane chess Problem Description   Hzz loves ...

  5. 概率DP入门学习QAQ

    emmmm博客很多都烂尾了...但是没空写..先写一下正在学的东西好了 概率DP这东西每次考到都不会..听题解也是一脸懵逼..所以决定学习一下这个东东..毕竟NOIP考过...比什么平衡树实在多了QA ...

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

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

  7. POJ 3156 - Interconnect (概率DP+hash)

    题意:给一个图,有些点之间已经连边,现在给每对点之间加边的概率是相同的,问使得整个图连通,加边条数的期望是多少. 此题可以用概率DP+并查集+hash来做. 用dp(i,j,k...)表示当前的每个联 ...

  8. 概率dp小结

    好久之前学过,记得是一次亚洲区的前几天看了看概率dp,然后亚洲区就出了一道概率dp,当时虽然做上了,但是感觉有很多地方没懂,今天起早温习了一下,觉得很多地方茅塞顿开,果然学习的话早上效果最好了. 首先 ...

  9. hdu4405:概率dp

    题意: 总共有n+1个格子:0-n 初始情况下在 0号格子 每次通过掷骰子确定前进的格子数 此外 还有一些传送门可以瞬间从 u 点传送到 v 点(必须被传送) 求走到(或超过)n点总共需要掷多少次骰子 ...

随机推荐

  1. 【转载】Lua中实现类的原理

    原文地址 http://wuzhiwei.net/lua_make_class/ 不错,将metatable讲的很透彻,我终于懂了. --------------------------------- ...

  2. 【洛谷 P2763】 试题库问题(最大流)

    题目链接 6/23 这是网络流23题里我第一个没看题解自己写出来一遍过的.. 这题应该是最简单的模型了吧. 从源点向每个类型连一条流量为这个类型要的题数,再从每个类型向可以属于这个类型的所有试题连一条 ...

  3. nyoj 15 括号匹配(二) (经典dp)

    题目链接 描述 给你一个字符串,里面只包含"(",")","[","]"四种符号,请问你需要至少添加多少个括号才能使这些 ...

  4. HTTPS加密通信原理及数字证书系统

    https加密通信原理: 公钥私钥成对,公钥公之于众,私钥只有自己知道. 用公钥加密的信息只能由与之相对应的私钥解密. 甲给乙发送数据时,甲先用乙的公钥加密这段数据,再用自己的私钥对这段数据的特征数据 ...

  5. ThinkPHP3.1.3 整合 UEditor百度编辑器 图片上传

    第一步.前端模板实例化百度编辑器 <js file='__ROOT__/Data/UEditor/ueditor.config.js' /> <js file='__ROOT__/D ...

  6. java===java基础学习(8)---静态域与静态方法

    静态域:如果将域定义为static,每个类中只有一个这样的域.而每一个对象对于所有的实例域却都有自己的一份拷贝.例如,加入需要给每一个雇员赋予唯一的标识码.这里给的Employee类添加一个实例域id ...

  7. Vim中的键映射【转】

    转自:http://www.cnblogs.com/softwaretesting/archive/2011/09/28/2194515.html http://www.pythonclub.org/ ...

  8. php文件读取的问题

    PHP字符编码问题 首先说下字符编码问题,当我们给定路径后如果路径中包含中文,可能会出现问题,打印到屏幕则显示没问题, 但是读取文件会报错:readfile(E:/素玄文件/app历史版本/素玄ERP ...

  9. caffe solver.prototxt 生成

    from caffe.proto import caffe_pb2 s = caffe_pb2.SolverParameter() path='/home/xxx/data/' solver_file ...

  10. node起server--axios做前端请求----进行CORS--跨域请求

    CORS是一个W3C标准,全称是"跨域资源共享"(Cross-origin resource sharing). 它允许浏览器向跨源服务器,发出XMLHttpRequest请求,从 ...