Solution -「CF 494C」Helping People
Link.
给定序列 和 个操作,第 个操作有 的概率将 内的元素 。且保证任意两个区间要么不交,要么有包含关系。求所有操作完成后序列最大值的期望。
,。
首先应当知道,(不然还需要做嘛 qwq),这是由于每个数的期望值是不独立的。
从题目奇怪的限制入手——各区间构成树形关系,整个序列上的区间构成一片森林。不妨加入第 个操作区间,满足 ,区间就构成一棵严格的树了。
考虑树上 DP,令 表示操作完 子树内的所有操作后,区间最大值 的概率。同时注意到 相较于值域大小 非常小,所以很多数是不可能成为最大值的。记 子树所代表的区间内初始元素的最大值 ,不难发现仅有 的 有意义,而其余 要不为 要不为 ,没有记录的必要。那么状态就能优化为操作完 子树内的所有操作后,区间最大值 的概率,并保证 。转移就简单了:
注意单独计算 ,因为其前一项应取 。
复杂度 。(前一项为预处理 ST 表复杂度。)
#include <cstdio>
#include <vector>
#include <algorithm>
const int MAXN = 1e5, MAXLG = 16, MAXM = 5000;
int n, m, mxa, a[MAXN + 5], lg[MAXN + 5], st[MAXN + 5][MAXLG + 5];
std::vector<int> tree[MAXM + 5];
double f[MAXM + 5][MAXM + 5];
inline void chkmax ( int& a, const int b ) { if ( a < b ) a = b; }
inline int min_ ( const int a, const int b ) { return a < b ? a : b; }
inline int rint () {
int x = 0; char s = getchar ();
for ( ; s < '0' || '9' < s; s = getchar () );
for ( ; '0' <= s && s <= '9'; s = getchar () ) x = x * 10 + ( s ^ '0' );
return x;
}
inline int qmax ( const int l, const int r ) {
int k = lg[r - l + 1], ret = st[l][k];
return chkmax ( ret, st[r - ( 1 << k ) + 1][k] ), ret;
}
struct Section {
int l, r, mx; double p;
inline void read () {
l = rint (), r = rint (), mx = qmax ( l, r );
scanf ( "%lf", &p );
}
inline bool operator < ( const Section t ) const {
return l ^ t.l ? l < t.l : r > t.r;
}
} sec[MAXM + 5];
inline void solve ( const int u ) {
for ( int v: tree[u] ) solve ( v );
f[u][0] = 1 - sec[u].p;
for ( int v: tree[u] ) f[u][0] *= f[v][sec[u].mx - sec[v].mx];
for ( int i = 1; i <= m; ++ i ) {
double p = 1, q = 1;
for ( int v: tree[u] ) {
p *= f[v][min_ ( sec[u].mx + i - sec[v].mx - 1, m )];
q *= f[v][min_ ( sec[u].mx + i - sec[v].mx, m )];
}
f[u][i] = sec[u].p * p + ( 1 - sec[u].p ) * q;
}
}
int main () {
n = rint (), m = rint ();
for ( int i = 1; i <= n; ++ i ) chkmax ( mxa, a[i] = st[i][0] = rint () );
for ( int i = 2; i <= n; ++ i ) lg[i] = lg[i >> 1] + 1;
for ( int j = 1; 1 << j <= n; ++ j ) {
for ( int i = 1; i + ( 1 << j ) - 1 <= n; ++ i ) {
chkmax ( st[i][j] = st[i][j - 1], st[i + ( 1 << j >> 1 )][j - 1] );
}
}
for ( int i = 1; i <= m; ++ i ) sec[i].read ();
sec[++ m] = { 1, n, qmax ( 1, n ), 0.0 };
std::sort ( sec + 1, sec + m + 1 );
for ( int i = 2; i <= m; ++ i ) {
for ( int j = i - 1; j; -- j ) {
if ( sec[j].l <= sec[i].l && sec[i].r <= sec[j].r ) {
tree[j].push_back ( i );
break;
}
}
}
solve ( 1 );
double ans = 0;
for ( int i = 0; i <= m; ++ i ) {
ans += ( i + mxa ) * ( f[1][i] - f[1][i - 1] );
}
printf ( "%.12f\n", ans );
return 0;
}
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 分享一个免费、快速、无限量使用的满血 DeepSeek R1 模型,支持深度思考和联网搜索!
· 基于 Docker 搭建 FRP 内网穿透开源项目(很简单哒)
· ollama系列01:轻松3步本地部署deepseek,普通电脑可用
· 25岁的心里话
· 按钮权限的设计及实现