Memory and Casinos CodeForces - 712E (概率,线段树)
题目大意:$n$个点, 每个点$i$有成功率$p_i$, 若成功走到$i+1$, 否则走到走到$i-1$, 多组询问, 求从$l$出发, 在$l$处不失败, 最后在$r$处胜利的概率
设$L[l,r]$表示从$l$出发, 在$l$处不失败, 最后在$r$处胜利的概率,$R[l,r]$表示从$r$出发, 在$l$处不失败, 最后在$r$处胜利的概率
记$l_1=L[l,mid],r_1=L[mid+1,r],l_2=R[l,mid],r_2=R[mid+1,r].$
枚举在$mid$到$mid+1$之间往返的次数, 可以得到下式
$$L[l,r]=l_1l_2\sum\limits_{n=0}^\infty{(1-l_2)^nr_1^n}=\frac{l_1l_2}{1-(1-l_2)r_1}$$
$$R[l,r]=r_2+(1-r_2)r_1l_2\sum\limits_{n=0}^\infty{(1-l_2)^nr_1^n}$$
$$=r_2+(1-r_2)\frac{r_1l_2}{1-(1-l_2)r_1}$$
直接用线段树维护就好了
还有另外一种做法, 考虑询问$[l,r]$
设$f_i$表示从$i$走到$r$且在$r$处胜利的概率, 则答案即为$f_l$
规定$f_{l-1}=0,f_{r+1}=1$ 就有$f_i=(1-p_i)f_{i-1}+p_if_{i+1}$
即$f_i-f_{i-1}=p_i(f_{i+1}-f_{i-1})$
令$g_i=f_i-f_{i-1}$ 则有$g_i=p_i(g_{i+1}-g_{i})$
记$a_i=\frac{1}{p_i}-1,$ 可以得到$g_{i+1}=a_ig_i$
由于$\sum_{i=l}^{r+1}{g_i}=1$
所以就可以得到$f_l=g_l=\frac{1}{1+a_l+a_la_{l+1}+...+a_la_{l+1}\cdot \cdot \cdot a_{r}}$
记$A[l,r]=\prod_{i=l}^{r}{a_i}, B[l,r]=\sum_{i=l}^{r}{A_i}$
有$$A[l,r]=A[l,mid]A[mid+1,r]$$
$$B[l,r]=B[l,mid]+A[l,mid]B[mid+1,r]$$
线段树维护$A,B$即可
第一种代码
#include <iostream>
#include <algorithm>
#include <math.h>
#include <cstdio>
#define REP(i,a,n) for(int i=a;i<=n;++i)
#define mid ((l+r)>>1)
#define lc (o<<1)
#define rc (lc|1)
#define ls lc,l,mid
#define rs rc,mid+1,r
using namespace std; const int N = 1e5+;
int n, m;
struct _ {
double L, R;
_ () {}
_ (int x, int y) {L=R=(double)x/y;}
_ (double l,double r) :L(l),R(r) {}
_ operator + (const _ &rhs) const {
return _(L*rhs.L/(-(-rhs.L)*R),rhs.R+(-rhs.R)*R*rhs.L/(-(-rhs.L)*R));
}
} v[N<<]; void build(int o, int l, int r) {
if (l==r) {
int x, y;
scanf("%d%d", &x, &y);
v[o]=_(x,y);
return;
}
build(ls),build(rs),v[o]=v[lc]+v[rc];
} void upd(int o, int l, int r, int pos, int x, int y) {
if (l==r) return v[o]=_(x,y),void();
if (mid>=pos) upd(ls,pos,x,y);
else upd(rs,pos,x,y);
v[o]=v[lc]+v[rc];
} _ qry(int o, int l, int r, int ql, int qr) {
if (ql<=l&&r<=qr) return v[o];
if (mid<ql) return qry(rs,ql,qr);
if (mid>=qr) return qry(ls,ql,qr);
return qry(ls,ql,qr)+qry(rs,ql,qr);
} int main() {
scanf("%d%d", &n, &m);
build(,,n);
REP(i,,m) {
int op, pos, a, b;
scanf("%d", &op);
if (op==) {
scanf("%d%d%d", &pos, &a, &b);
upd(,,n,pos,a,b);
} else {
scanf("%d%d", &a, &b);
printf("%.12lf\n", qry(,,n,a,b).L);
}
}
}
第二种完全类似, 改下公式就行了
#include <iostream>
#include <algorithm>
#include <math.h>
#include <cstdio>
#define REP(i,a,n) for(int i=a;i<=n;++i)
#define mid ((l+r)>>1)
#define lc (o<<1)
#define rc (lc|1)
#define ls lc,l,mid
#define rs rc,mid+1,r
using namespace std; const int N = 1e5+;
int n, m;
struct _ {
double A, B;
_ () {}
_ (int x, int y) {A=B=(double)y/x-;}
_ (double a,double b) :A(a),B(b) {}
_ operator + (const _ &rhs) const {
return _(A*rhs.A,B+A*rhs.B);
}
} v[N<<]; void build(int o, int l, int r) {
if (l==r) {
int x, y;
scanf("%d%d", &x, &y);
v[o]=_(x,y);
return;
}
build(ls),build(rs),v[o]=v[lc]+v[rc];
} void upd(int o, int l, int r, int pos, int x, int y) {
if (l==r) return v[o]=_(x,y),void();
if (mid>=pos) upd(ls,pos,x,y);
else upd(rs,pos,x,y);
v[o]=v[lc]+v[rc];
} _ qry(int o, int l, int r, int ql, int qr) {
if (ql<=l&&r<=qr) return v[o];
if (mid<ql) return qry(rs,ql,qr);
if (mid>=qr) return qry(ls,ql,qr);
return qry(ls,ql,qr)+qry(rs,ql,qr);
} int main() {
scanf("%d%d", &n, &m);
build(,,n);
REP(i,,m) {
int op, pos, a, b;
scanf("%d", &op);
if (op==) {
scanf("%d%d%d", &pos, &a, &b);
upd(,,n,pos,a,b);
} else {
scanf("%d%d", &a, &b);
printf("%.12lf\n", /(+qry(,,n,a,b).B));
}
}
}
Memory and Casinos CodeForces - 712E (概率,线段树)的更多相关文章
- Codeforces 787D. Legacy 线段树建模+最短路
D. Legacy time limit per test:2 seconds memory limit per test:256 megabytes input:standard input out ...
- CodeForces 91B Queue (线段树,区间最值)
http://codeforces.com/problemset/problem/91/B B. Queue time limit per test: 2 seconds memory limit p ...
- Vasya and a Tree CodeForces - 1076E(线段树+dfs)
I - Vasya and a Tree CodeForces - 1076E 其实参考完别人的思路,写完程序交上去,还是没理解啥意思..昨晚再仔细想了想.终于弄明白了(有可能不对 题意是有一棵树n个 ...
- loj2537 「PKUWC2018」Minimax 【概率 + 线段树合并】
题目链接 loj2537 题解 观察题目的式子似乎没有什么意义,我们考虑计算出每一种权值的概率 先离散化一下权值 显然可以设一个\(dp\),设\(f[i][j]\)表示\(i\)节点权值为\(j\) ...
- CodeForces 19D Points (线段树+set)
D. Points time limit per test 2 seconds memory limit per test 256 megabytes input standard input out ...
- Codeforces 240F. TorCoder 线段树
线段树统计和维护某一区间内的字母个数.. . . F. TorCoder time limit per test 3 seconds memory limit per test 256 megabyt ...
- Almost Regular Bracket Sequence CodeForces - 1095E (线段树,单点更新,区间查询维护括号序列)
Almost Regular Bracket Sequence CodeForces - 1095E You are given a bracket sequence ss consisting of ...
- Sereja and Brackets CodeForces - 380C (线段树+分治思路)
Sereja and Brackets 题目链接: CodeForces - 380C Sereja has a bracket sequence s1, s2, ..., *s**n, or, in ...
- Glass Carving CodeForces - 527C (线段树)
C. Glass Carving time limit per test2 seconds memory limit per test256 megabytes inputstandard input ...
随机推荐
- Python 面试题集锦【315+道题】
第一部分 Python基础篇(80题) 为什么学习Python? 通过什么途径学习的Python? Python和Java.PHP.C.C#.C++等其他语言的对比? 简述解释型和编译型编程语言? P ...
- 【封装函数】原生js 获取行内外联样式-兼容IE
var dom=document.getElementsByTagName("div")[0]; console.log(getStyle(dom,"padding-to ...
- Filter过滤器原理和登录实现
Filter过滤器API Servlet过滤器API包含了3个接口,它们都在javax.servlet包中,分别是Filter接口.FilterChain接口和FilterConfig接口. ...
- Linux学习笔记之如何让普通用户获得ROOT权限
在学习sodu的时候,我发现一些命令只能由root用户使用,普通用户使用会提示此用户没有使用sudo的权限.我想到的解方法是把正在使用的普通用户获得root权限,于是我通过百度和询问老师知道了如何去实 ...
- win10锁屏界面无法更新
win10的锁屏界面都是巨硬公司推送过来的,质量还不错,最近锁屏界面无法更新,解决方案如下: 以管理员身份运行cmd,分别运行如下两个命令 del /f /s /q /a "%userpro ...
- 20145127《java程序设计》第四周学习总结
教材学习内容总结 第六章 继承与多态 6.1 何为继承 0.面向对象中,子类继承父类,避免城府的行为定义.正确判断使用继承的时机,以及继承之后如何活用多态,才是学习继承时的重点. 1.继承:避免多个类 ...
- CDC画图
CDC* pdc: CRect rcBounds: 1. 画直线 pdc->MoveTo(rcBounds.TopLeft());//将画笔移动到左上角这个点,使用这个点作为起点画图 pdc-& ...
- Linux内核分析--内核中的数据结构双向链表续【转】
在解释完内核中的链表基本知识以后,下面解释链表的重要接口操作: 1. 声明和初始化 实际上Linux只定义了链表节点,并没有专门定义链表头,那么一个链表结构是如何建立起来的呢?让我们来看看LIST_H ...
- SpringBoot中使用mybatis-generator自动生产
步骤: 1.在pom.xml中添加插件配置 <plugin> <groupId>org.mybatis.generator</groupId> <artifa ...
- 筛选出sql 查询结果中 不包含某个字符
select * from table1 where patindex('%关键字%' , aa) = 0 select * from table1 where charindex('关键字' , a ...