Comet OJ - Contest #11 D isaster 重构树+倍增+dfs序+线段树
发现对于任意一条边,起决定性作用的是节点编号更大的点.
于是,对于每一条边,按照节点编号较大值作为边权,按照最小生成树的方式插入即可.
最后用线段树维护 dfs 序做一个区间查询即可.
Code:
#include <bits/stdc++.h>
#define N 400005
#define ll long long
#define mod 998244353
#define setIO(s) freopen(s".in","r",stdin)
using namespace std;
ll val[N],mul[N<<2],V[N];
int n,m,Q,tot,edges,tt;
int p[N],cc[N],hd[N],to[N],nex[N],dfn[N],st[N],ed[N],size[N],Fa[21][N],F[21][N];
struct Edge
{
int u,v,c;
Edge(int u=0,int v=0,int c=0):u(u),v(v),c(c){}
}e[N];
void pushup(int l,int r,int now)
{
ll re=1;
int mid=(l+r)>>1;
if(l<=mid)re=re*mul[now<<1]%mod;
if(r>mid) re=re*mul[now<<1|1]%mod;
mul[now]=re;
}
void build(int l,int r,int now)
{
if(l==r)
{
mul[now]=V[l]%mod;
return;
}
int mid=(l+r)>>1;
if(l<=mid) build(l,mid,now<<1);
if(r>mid) build(mid+1,r,now<<1|1);
pushup(l,r,now);
}
void update(int l,int r,int now,int p,int v)
{
if(l==r)
{
V[l]=mul[now]=1ll*v%mod;
return;
}
int mid=(l+r)>>1;
if(p<=mid) update(l,mid,now<<1,p,v);
else update(mid+1,r,now<<1|1,p,v);
pushup(l,r,now);
}
ll query(int l,int r,int now,int L,int R)
{
if(l>=L&&r<=R) return mul[now];
ll re=1;
int mid=(l+r)>>1;
if(L<=mid) re=re*query(l,mid,now<<1,L,R)%mod;
if(R>mid) re=re*query(mid+1,r,now<<1|1,L,R)%mod;
return re;
}
int find(int x)
{
return p[x]==x?x:p[x]=find(p[x]);
}
bool cmp(Edge a,Edge b)
{
return a.c<b.c;
}
void addedge(int u,int v)
{
nex[++edges]=hd[u],hd[u]=edges,to[edges]=v;
}
void dfs(int u,int ff)
{
st[u]=dfn[u]=++tt, size[u]=1, V[dfn[u]]=val[u];
Fa[0][u]=ff, F[0][u]=max(cc[ff], cc[u]);
for(int i=1;i<=20;++i)
{
Fa[i][u]=Fa[i-1][Fa[i-1][u]];
F[i][u]=max(F[i-1][u], F[i-1][Fa[i-1][u]]);
}
for(int i=hd[u];i;i=nex[i])
{
int v=to[i];
dfs(v, u);
size[u]+=size[v];
}
ed[u]=tt;
}
int get(int x,int k)
{
for(int i=20;i>=0;--i)
if(Fa[i][x] && F[i][x] <= k) x=Fa[i][x];
return x;
}
int main()
{
int i,j,k;
// setIO("input");
scanf("%d%d%d",&n,&m,&Q);
tot=n;
for(i=1;i<N;++i) p[i]=i;
for(i=1;i<=n;++i) scanf("%lld",&val[i]),cc[i]=i;
for(i=1;i<=m;++i)
{
scanf("%d%d",&e[i].u,&e[i].v);
e[i].c=max(e[i].u, e[i].v);
}
sort(e+1,e+1+m,cmp);
for(i=1;i<=m;++i)
{
int u=e[i].u, v=e[i].v;
int x=find(u),y=find(v);
if(x!=y)
{
++tot;
val[tot]=1ll;
cc[tot]=e[i].c;
p[x]=tot,p[y]=tot;
addedge(tot,x), addedge(tot,y);
}
}
cc[0]=1000000000;
dfs(tot,0);
build(1,tot,1);
for(i=1;i<=Q;++i)
{
int op;
scanf("%d",&op);
if(op==1)
{
int x,y;
scanf("%d%d",&x,&y);
if(x>y) printf("0\n");
else
{
int k=get(x,y);
int L=st[k], R=ed[k];
printf("%lld\n",query(1,tot,1,L,R));
}
}
if(op==2)
{
int x,y;
scanf("%d%d",&x,&y);
update(1,tot,1,dfn[x],y);
}
}
return 0;
}
Comet OJ - Contest #11 D isaster 重构树+倍增+dfs序+线段树的更多相关文章
- isaster(Comet OJ - Contest #11D题+kruskal重构树+线段树+倍增)
目录 题目链接 思路 代码 题目链接 传送门 思路 \(kruskal\)重构树\(+\)线段树\(+\)倍增 代码 #include <set> #include <map> ...
- Comet OJ - Contest #11 题解&赛后总结
Solution of Comet OJ - Contest #11 A.eon -Problem designed by Starria- 在模 10 意义下,答案变为最大数的最低位(即原数数位的最 ...
- Comet OJ - Contest #11题解
传送门 \(A\) 咕咕咕 const int N=1e6+5; char s[N],t[N];int n,res; inline bool cmp(const int &x,const in ...
- 2017 ACM-ICPC, Universidad Nacional de Colombia Programming Contest K - Random Numbers (dfs序 线段树+数论)
Tamref love random numbers, but he hates recurrent relations, Tamref thinks that mainstream random g ...
- Comet OJ - Contest #11 B题 usiness
###题目链接### 题目大意:一开始手上有 0 个节点,有 n 天抉择,m 种方案,在每天中可以选择任意种方案.任意次地花费 x 个节点(手上的节点数不能为负),使得在 n 天结束后,获得 y 个节 ...
- Comet OJ - Contest #11 B 背包dp
Code: #include <bits/stdc++.h> #define N 1005 #define M 2000 #define setIO(s) freopen(s". ...
- Comet OJ - Contest #11 A 水题
Code: #include <bits/stdc++.h> #define N 3000000 using namespace std; char str[N]; int main() ...
- Comet OJ - Contest #11 E ffort(组合计数+多项式快速幂)
传送门. 题解: 考虑若最后的总伤害数是s,那么就挡板分配一下,方案数是\(C_{s-1}^{n-1}\). 那么问题在于总伤害数很大,不能一个一个的算. \(C_{s-1}^{n-1}\)的OGF是 ...
- Comet OJ - Contest #2 简要题解
Comet OJ - Contest #2 简要题解 cometoj A 模拟,复杂度是对数级的. code B 易知\(p\in[l,r]\),且最终的利润关于\(p\)的表达式为\(\frac{( ...
随机推荐
- 剑指offer18:操作给定的二叉树,将其变换为源二叉树的镜像。
1 题目描述 操作给定的二叉树,将其变换为源二叉树的镜像. 2 输入描述: 二叉树的镜像定义:源二叉树 8 / \ 6 10 / \ / \ 5 7 9 11 镜像二叉树 8 / \ 10 6 / \ ...
- 请写一段 PHP 代码 ,确保多个进程同时写入同一个文件成功
方案一: function writeData($filepath, $data) { $fp = fopen($filepath,'a'); do{ usleep(100); }while (!fl ...
- Thinkphp自定义生成缩略图尺寸的方法
Thinkphp自定义生成缩略图尺寸的方法,本实例中生成两张不同尺寸的图片:第一张是大图350*350,第二张 50*50的缩略图 Image类是Thinkphp系统自带的,可以研究下,这个缩略图类很 ...
- fiddler笔记:web session窗口介绍
1.web session列表的含义:(从左到右) # fiddler通过session生成的ID. Result 响应状态码. Host 接收请求的服务器的主机名和端口号. URL 请求资源的位置. ...
- 用shell脚本安装MySQL-5.7.22-Percona版本
#!/bin/bash MySQL_Package=Percona-Server-5.7.22-22-Linux.x86_64.ssl101.tar.gz Package_Source=Percona ...
- 第十一章 ZYNQ-MIZ701 PS读写PL端BRAM
本篇文章目的是使用Block Memory进行PS和PL的数据交互或者数据共享,通过zynq PS端的Master GP0端口向BRAM写数据,然后再通过PS端的Mater GP1把数据读出来,将 ...
- 洛谷P1087 FBI树
P1087 FBI树题解: 看到这个题,我想到了线段树!(毕竟刚搞完st表...) 当然,题解中有位大佬也用的线段树,但是当时看的时候我看见了9个if,当场去世. 那么这是一个不用暴力的线段树,且简单 ...
- X86驱动:挂接SSDT内核钩子
SSDT 中文名称为系统服务描述符表,该表的作用是将Ring3应用层与Ring0内核层,两者的API函数连接起来,起到承上启下的作用,SSDT并不仅仅只包含一个庞大的地址索引表,它还包含着一些其它有用 ...
- react开发vscode插件推荐
原文地址:https://github.com/xieqingtian/blog/issues/2 由于本人主要是做react开发,用的代码编辑器是传说中的宇宙第一前端神器vscode, 所以在这里记 ...
- input type=color 设置颜色
在设置背景色的时候,使用html5 type=color 标签,但是初始值一直都是黑色的,背景如果没有设置的时候,应该是白色,比如文本图元,所以需要设置一个初始的颜色值, 注意: value不实用,怎 ...