题目链接

题目大意:$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 (概率,线段树)的更多相关文章

  1. Codeforces 787D. Legacy 线段树建模+最短路

    D. Legacy time limit per test:2 seconds memory limit per test:256 megabytes input:standard input out ...

  2. CodeForces 91B Queue (线段树,区间最值)

    http://codeforces.com/problemset/problem/91/B B. Queue time limit per test: 2 seconds memory limit p ...

  3. Vasya and a Tree CodeForces - 1076E(线段树+dfs)

    I - Vasya and a Tree CodeForces - 1076E 其实参考完别人的思路,写完程序交上去,还是没理解啥意思..昨晚再仔细想了想.终于弄明白了(有可能不对 题意是有一棵树n个 ...

  4. loj2537 「PKUWC2018」Minimax 【概率 + 线段树合并】

    题目链接 loj2537 题解 观察题目的式子似乎没有什么意义,我们考虑计算出每一种权值的概率 先离散化一下权值 显然可以设一个\(dp\),设\(f[i][j]\)表示\(i\)节点权值为\(j\) ...

  5. CodeForces 19D Points (线段树+set)

    D. Points time limit per test 2 seconds memory limit per test 256 megabytes input standard input out ...

  6. Codeforces 240F. TorCoder 线段树

    线段树统计和维护某一区间内的字母个数.. . . F. TorCoder time limit per test 3 seconds memory limit per test 256 megabyt ...

  7. Almost Regular Bracket Sequence CodeForces - 1095E (线段树,单点更新,区间查询维护括号序列)

    Almost Regular Bracket Sequence CodeForces - 1095E You are given a bracket sequence ss consisting of ...

  8. Sereja and Brackets CodeForces - 380C (线段树+分治思路)

    Sereja and Brackets 题目链接: CodeForces - 380C Sereja has a bracket sequence s1, s2, ..., *s**n, or, in ...

  9. Glass Carving CodeForces - 527C (线段树)

    C. Glass Carving time limit per test2 seconds memory limit per test256 megabytes inputstandard input ...

随机推荐

  1. Java EE业务处理流程与XML的引入

    Java EE基于MVC架构的业务处理流程 MVC架构业务处理流程 XML定义 XML是可扩展标记语言,标准通用标记语言的子集,是一种用于标记电子文件使其具有结构性的标记语言.XML被设计用于数据的存 ...

  2. 用rewrite把旧域名直接跳转到新域名的nginx配置

    用rewrite把旧域名直接跳转到新域名的nginx配置 把下面代码保存到daziran.com.conf 放在nginx配置目录下 /etc/nginx/conf.d/ #把旧域名zdz8207直接 ...

  3. centos上安装python3.6

    安装python3.6可能使用的依赖 # yum install openssl-devel bzip2-devel expat-devel gdbm-devel readline-devel sql ...

  4. 手撕vue-cli配置文件——check-versions.js篇

    check-versions.js,vue-cli中检查版本的js文件. 'use strict' const chalk = require('chalk') const semver = requ ...

  5. tomcat 7下spring 4.x mvc集成websocket以及sockjs完全参考指南(含nginx/https支持)

    之所以sockjs会存在,说得不好听点,就是因为微软是个流氓,现在使用windows 7的系统仍然有近半,而windows 7默认自带的是ie 8,有些会自动更新到ie 9,但是大部分非IT用户其实都 ...

  6. Arduino使用HC05蓝牙模块与手机连接

    Arduino使用HC05蓝牙模块与手机连接 一切都是最好的选择 首先是线路连接,一定不要接错了 Arduino 代码 #include <SoftwareSerial.h> // Pin ...

  7. 【翻译】std::list::remove - C++ Reference

    公有成员函数 std::list::remove void remove(const value_type& val); 删除与给定值相等的元素 从容器中删除所有与 val 值相等的元素.li ...

  8. 【第八章】 springboot + mybatis + 多数据源

    在实际开发中,我们一个项目可能会用到多个数据库,通常一个数据库对应一个数据源. 代码结构: 简要原理: 1)DatabaseType列出所有的数据源的key---key 2)DatabaseConte ...

  9. 51Nod 1091 线段的重叠(贪心+区间相关

    1091 线段的重叠 X轴上有N条线段,每条线段包括1个起点和终点.线段的重叠是这样来算的,[10 20]和[12 25]的重叠部分为[12 20]. 给出N条线段的起点和终点,从中选出2条线段,这两 ...

  10. Java ArrayDeque源码剖析

    ArrayDeque 本文github地址 前言 Java里有一个叫做Stack的类,却没有叫做Queue的类(它是个接口名字).当需要使用栈时,Java已不推荐使用Stack,而是推荐使用更高效的A ...