感觉期望DP这种东西像是玄学…

主要总结说一点基础性的东西, 或许对于理解题目的做法会有一点帮助.

首先是关于独立事件, 互斥事件的概念. 通俗地说, 就是对于两个事件A, B, 假如满足发生了其中一个就不会发生另一个, 则称A, B为互斥事件; 假如A与B的发生没有任何关系, 可能都发生, 可能只有一个发生, 也可能都不发生, 则称A, B为独立事件. 下文的讨论主要针对独立事件进行.

一. 事件间的关系与运算;

A + B (和事件): 表示A, B两个事件至少有一个发生;

A · B(积事件)表示A, B两个事件同时发生;

二. 复杂事件的概率运算公式

和事件的概率: P(A + B) = P(A) + P(B) - P(A * B)

推广得到: P(A)=(P(A+B)−P(B))/(1−P(B))

即: P(A - B) = (P(A) - P(B)) / (1 - P(B))

适用于在知道和事件与其中一个事件的概率的情况下, 求另一个事件的概率.

特别地, 当A与B为互斥事件时, P(A + B) = P(A) + P(B)

积事件的概率: P(A1 * A2 * A3 * … * An) = P(A1) * P(A2) * … * P(An)

三. 期望值的计算公式:

Ex = ΣXiPi(ΣPi = 1)

例题: BZOJ3566概率充电器

这一题就用到了和公式与差公式

#include<cstdio>
#include<cctype>
#include<cstring>
using namespace std;
inline int read()
{
int x = 0, flag = 1;
char c;
while(! isgraph(c = getchar()))
if(c == '-')
flag *= - 1;
while(isgraph(c))
x = x * 10 + c - '0', c = getchar();
return x * flag;
}
void println(int x)
{
if(x < 0)
putchar('-'), x *= - 1;
if(x == 0)
putchar('0');
int top = 0, ans[1 << 4];
while(x)
ans[top ++] = x % 10, x /= 10;
for(; top; top --)
putchar(ans[top - 1] + '0');
putchar('\n');
}
const double EPS = 1e-8;
const int MAXN = 1 << 20;
int head[MAXN];
int top;
struct edge
{
int v, next;
double p;
}T[MAXN << 1];
void add_edge(int u, int v, double p)
{
T[top].v = v, T[top].next = head[u], T[top].p = p, head[u] = top ++;
}
double q[MAXN];
double f[MAXN], ans[MAXN];
void DFS(int u, int fa)
{
f[u] = q[u];
for(int i = head[u]; i != - 1; i = T[i].next)
{
int v = T[i].v;
if(v == fa)
continue;
DFS(v, u);
f[u] = f[u] + f[v] * T[i].p - f[u] * f[v] * T[i].p;
}
}
void DGS(int u, int fa)
{
for(int i = head[u]; i != - 1; i = T[i].next)
{
int v = T[i].v;
if(v == fa)
continue;
if(1 - f[v] * T[i].p < EPS)
ans[v] = 1.0;
else
{
double x = (ans[u] - f[v] * T[i].p) / (1 - f[v] * T[i].p) * T[i].p;
ans[v] = f[v] + x - f[v] * x;
}
DGS(v, u);
}
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("BZOJ3566.in", "r", stdin);
freopen("BZOJ3566.out", "w", stdout);
#endif
int n = read();
memset(head, - 1, sizeof(head));
top = 0;
for(int i = 1; i < n; i ++)
{
int a = read(), b = read(), p = read();
add_edge(a, b, (double)p / 100.0);
add_edge(b, a, (double)p / 100.0);
}
for(int i = 1; i <= n; i ++)
q[i] = (double)read() / 100.0;
memset(f, 0, sizeof(f));
DFS(1, 1);
ans[1] = f[1];
DGS(1, 1);
double sum = 0;;
for(int i = 1; i <= n; i ++)
sum += ans[i];
printf("%.6f", sum);
}


最新文章

  1. 在.net中调用Delphi dll的Pchar转换
  2. Array数组标准库
  3. IIS------IIS上部署MVC网站,打开后ExtensionlessUrlHandler-Integrated-4.0解决办法
  4. 学习Sass之安装Sass(一)
  5. Chrome使用技巧(几个月的心得)
  6. Unity4.5版本DLL库名字问题
  7. docker confluence
  8. RESTful架构详解(转)
  9. FluorineFx 播放FLV 时堆棧溢出解决 FluorineFx NetStream.play 并发时,无法全部连接成功的解决办法
  10. Oracle数据库小知识,改数据库数据
  11. python学习之list
  12. 【canvas系列】canvas实现“ 简单的Amaziograph效果”--画对称图
  13. Selenium元素定位之Xpath
  14. 缓冲区(buffer)与缓存(cache)
  15. 常见web UI 元素操作 及API使用
  16. js返回值 数组去重
  17. UVA-714-二分+贪心
  18. 关于oracle的基础增删改查操作总结
  19. EXCEPTION:FATAL: UNABLE TO CREATE ‘…GIT/INDEX.LOCK’ FILE EXISTS
  20. 使用VI编辑器在Linux下编写Java文件

热门文章

  1. Linux学习-函式库管理
  2. myeclipse中hibernate生成映射文件
  3. zuul sample
  4. pg 创建自增id
  5. C语言变量长度在32位和64位处理器上的关系
  6. [UiAutomator篇][3] 打开音乐应用的测试脚本
  7. List容器——LinkedList及常用API,实现栈和队列
  8. Can not issue data manipulation statements with executeQuery().解决方案
  9. Linux Shell系列教程之(四)Shell注释
  10. win install pip