「ZJOI2019」线段树 解题报告
「ZJOI2019」线段树
听说有人喷这个题简单,然后我就跑去做,然后自闭感++,rp++(雾)
理性分析一波,可以发现最后形成的\(2^k\)个线段树,对应的操作的一个子集,按时间顺序作用到这颗线段树上。
首先考虑研究一下tag的性质,比如两个操作时间先后是否没有影响,操作是否可以以某种形式进行合并,然后啥也没发现。
然后考虑一下一颗树是否可以被压成某个状态,比如实际上只有\(\log\)个状态然后去dp,发现也不行
再次冷静分析一波,发现好像每个节点可以独立考虑,结合上面\(2^n\),不妨考虑转换成概率,算出每个点被打上tag的概率,在数值上同时也是它的期望,这样我们最后乘上线段树个数就好了。
然后这时候就只对一个线段树操作了,考虑咋去搞
这差不多是我做题时的心路历程,但是我写了好几个做法都过不了大样例,十分自闭,总是以为自己没想清楚哪里假了之类的赶紧改过来,结果大样例输出了十几个不同的答案
心态崩了.jpg
又被什么pkuthusc的报名表填的心烦意乱的,就直接去看题解了
发现自己压根没有对节点分类讨论的意识...
设\(dp_i\)表示\(i\)节点被打上标记的概率

考虑对区间\([4,6]\)进行操作后,节点的改变。
后文中,原来的一半因为不变,所以节点改变指变化的那一半的改变。
红色点
有标记也全部被push了
\[dp_i=\frac{dp_i}{2}
\]绿色点
一定被打上标记了
\[dp_i=\frac{dp_i+1}{2}
\]黑色点
考虑它到跟的红色点是不是有一个点给它放标记了,或者原本是它自己的
这个没法直接搞,定义一个辅助数组\(f_i\)表示\(i\)到根至少有一个点有标记的概率
那么有
\[dp_i=\frac{dp_i+f_i}{2}
\]白色点不变
那考虑球一下辅助数组\(f\)
红色点
直接没了
\[f_i=\frac{f_i}{2}
\]绿色点
发现绿色点的子树,都有了一个绿色点,即
\[f_i=\frac{f_i+1}{2}
\]需要实现一个子树加
我们发现形如\(x\leftarrow\frac{x+1}{2}\)的式子,如果进行\(n\)次,会变成\(\frac{x+2^n-1}{2^n}\)
于是可以直接打标记记录一下次数
灰色点
发现有就放到灰色点,没有仍然没有,所以灰色点的子树不变
总复杂度\(O(n\log n)\)
Code:
#include <cstdio>
#include <cctype>
const int SIZE=1<<21;
char ibuf[SIZE],*iS,*iT;
#define gc() (iS==iT?(iT=(iS=ibuf)+fread(ibuf,1,SIZE,stdin),iS==iT?EOF:*iS++):*iS++)
template <class T>
void read(T &x)
{
x=0;char c=gc();
while(!isdigit(c)) c=gc();
while(isdigit(c)) x=x*10+c-'0',c=gc();
}
const int mod=998244353;
const int inv2=499122177;
const int N=1e5+10;
#define mul(a,b) (1ll*(a)*(b)%mod)
inline int add(int a,int b){return a+b>=mod?a+b-mod:a+b;}
#define ls id<<1
#define rs id<<1|1
int fdp[N<<2],dp[N<<2],sum[N<<2],tag[N<<2];
int n,m,in2[N];
void updata(int id)
{
sum[id]=add(add(sum[ls],sum[rs]),dp[id]);
}
void pushdown(int id)
{
tag[ls]+=tag[id];
tag[rs]+=tag[id];
fdp[ls]=add(mul(fdp[ls],in2[tag[id]]),add(1,mod-in2[tag[id]]));
fdp[rs]=add(mul(fdp[rs],in2[tag[id]]),add(1,mod-in2[tag[id]]));
tag[id]=0;
}
void modify(int id,int L,int R,int l,int r)
{
if(l==L&&r==R)
{
sum[id]=add(sum[id],mod-dp[id]);
dp[id]=mul(dp[id]+1,inv2);
fdp[id]=mul(fdp[id]+1,inv2);
sum[id]=add(sum[id],dp[id]);
++tag[id];
return;
}
dp[id]=mul(dp[id],inv2);
fdp[id]=mul(fdp[id],inv2);
pushdown(id);
int Mid=L+R>>1;
if(r<=Mid)
{
sum[rs]=add(sum[rs],mod-dp[rs]);
dp[rs]=mul(add(dp[rs],fdp[rs]),inv2);
sum[rs]=add(sum[rs],dp[rs]);
modify(ls,L,Mid,l,r);
}
else if(l>Mid)
{
sum[ls]=add(sum[ls],mod-dp[ls]);
dp[ls]=mul(add(dp[ls],fdp[ls]),inv2);
sum[ls]=add(sum[ls],dp[ls]);
modify(rs,Mid+1,R,l,r);
}
else
modify(ls,L,Mid,l,Mid),modify(rs,Mid+1,R,Mid+1,r);
updata(id);
}
int main()
{
read(n),read(m);
in2[0]=1;
for(int i=1;i<=n;i++) in2[i]=mul(in2[i-1],inv2);
for(int t=1,op,l,r,i=1;i<=m;i++)
{
read(op);
if(op==1)
{
read(l),read(r);
modify(1,1,n,l,r);
t=add(t,t);
}
else printf("%lld\n",mul(sum[1],t));
}
return 0;
}
2019.5.10
「ZJOI2019」线段树 解题报告的更多相关文章
- 【LOJ】#3043. 「ZJOI2019」线段树
LOJ#3043. 「ZJOI2019」线段树 计数转期望的一道好题-- 每个点设两个变量\(p,q\)表示这个点有\(p\)的概率有标记,有\(q\)的概率到祖先的路径上有个标记 被覆盖的点$0.5 ...
- 「SHOI2014」三叉神经树 解题报告
「SHOI2014」三叉神经树 膜拜神仙思路 我们想做一个类似于动态dp的东西,首先得确保我们的运算有一个交换律,这样我们可以把一长串的运算转换成一块一块的放到矩阵上之类的东西,然后拿数据结构维护. ...
- LOJ 3043: 洛谷 P5280: 「ZJOI2019」线段树
题目传送门:LOJ #3043. 题意简述: 你需要模拟线段树的懒标记过程. 初始时有一棵什么标记都没有的 \(n\) 阶线段树. 每次修改会把当前所有的线段树复制一份,然后对于这些线段树实行一次区间 ...
- 「ZJOI2019」线段树
传送门 Description 线段树的核心是懒标记,下面是一个带懒标记的线段树的伪代码,其中 tag 数组为懒标记: 其中函数\(Lson(Node)\)表示\(Node\)的左儿子,\(Rson( ...
- 【LOJ3043】「ZJOI2019」线段树
题面 问题可以转化为每次区间覆盖操作有 \(\frac{1}{2}\) 的概率进行,求标记和的期望.于是我们只要求出所有点有标记的概率即可. 我们设 \(f_i\) 表示节点 \(i\) 有标记的概率 ...
- @loj - 3043@「ZJOI2019」线段树
目录 @description@ @solution@ @accepted code@ @details@ @description@ 九条可怜是一个喜欢数据结构的女孩子,在常见的数据结构中,可怜最喜 ...
- 「模板」 线段树——区间乘 && 区间加 && 区间求和
「模板」 线段树--区间乘 && 区间加 && 区间求和 原来的代码太恶心了,重贴一遍. #include <cstdio> int n,m; long l ...
- Loj #2570. 「ZJOI2017」线段树
Loj #2570. 「ZJOI2017」线段树 题目描述 线段树是九条可怜很喜欢的一个数据结构,它拥有着简单的结构.优秀的复杂度与强大的功能,因此可怜曾经花了很长时间研究线段树的一些性质. 最近可怜 ...
- 「ZJOI2016」大森林 解题报告
「ZJOI2016」大森林 神仙题... 很显然线段树搞不了 考虑离线操作 我们只搞一颗树,从位置1一直往后移动,然后维护它的形态试试 显然操作0,1都可以拆成差分的形式,就是加入和删除 因为保证了操 ...
随机推荐
- 【leetcode】316. Remove Duplicate Letters
题目如下: Given a string which contains only lowercase letters, remove duplicate letters so that every l ...
- c++11引入特性
* 支持类内初始化. class A{ vector<string> strs{"abc", "def"}; };
- jstat性能分析
垃圾回收统计 S0C:第一个幸存区的大小 S1C:第二个幸存区的大小 S0U:第一个幸存区的使用大小 S1U:第二个幸存区的使用大小 EC:伊甸园区的大小 EU:伊甸园区的使用大小 OC:老年代大小 ...
- mac 格式化U盘
作者:Bailm链接:https://www.zhihu.com/question/27888608/answer/486347894来源:知乎著作权归作者所有.商业转载请联系作者获得授权,非商业转载 ...
- http协议和file协议的区别
1.在本地直接使用浏览器打开 html文件 和 通过本地服务器打开 html文件 有什么区别呢. https://segmentfault.com/q/1010000006554881/a-1 ...
- 2018icpc南京/gym101981 I Magic Potion
题意: 若干个勇士,每个勇士只能杀特定的怪物.每个勇士只能杀1个怪,但是有一些药,喝了药之后能再杀一个,每个勇士只能喝一瓶药.问你最多杀多少怪. 题解: 按照如下建图套网络流板即可. 网上有题解说套D ...
- kali开启禁止或删除ssh 开机启动
开启禁止或删除ssh 开机启动 # update-rc.d ssh enable #//开机启动 # update-rc.d ssh disable #//禁止开机启动 # update-rc.d - ...
- webservice 应用
一直以来,dashboard就会面临一个非常难堪的问题.就是刷新速度太慢了.它要连接query 来获取数据.而query每刷一次都需要时间.这是无可避免的结果.尽管它也是结果集,可还是比较慢.最近实践 ...
- c# networkcomms 3.0实现模拟登陆总结 转载https://www.cnblogs.com/zuochanzi/p/7039636.html
最近项目需要做一个客户查询状态系统,当前上位机缺少服务功能,于是找到了networkcomms 开源框架,作为项目使用. 最新版networkcomms 下载地址:https://github.com ...
- Django框架(十八)—— auth框架:用户登录、注册、认证
目录 auth模块 一.什么是author模块 二.auth模块的使用 1.创建超级用户(create_superuser()) 2.验证用户(authenticate()) 3.登录用户(login ...