[cf1137F]Matches Are Not a Child's Pla
2021/8/30 23:06:54
本文主要是介绍[cf1137F]Matches Are Not a Child's Pla,对大家解决编程问题具有一定的参考价值,需要的程序猿们随着小编来一起学习吧!
显然compare操作可以通过两次when操作实现,以下仅考虑前两种操作
为了方便,将优先级最高的节点作为根,显然根最后才会被删除
接下来,不断找到剩下的节点中(包括根)优先级最高的节点,将其到其所在树根的所有节点从下到上依次加入到序列的开头并删除,不难发现最终得到的序列即为燃烧的顺序
将每一次删除的链上的边称为实边,其余边称为虚边,实际上就构成了一个类似于LCT的结构
一次$v$的修改操作对该LCT的影响,简单分析后不难发现即为将$v$换为根
考虑节点$v$的答案,即分为两部分:
1.定义其中一条实链(一个Splay)的优先级为其中优先级最大的节点(链尾),所有实链中优先级比$v$所在实链小的长度(指节点个数)和
2.$v$所在实链中比$v$浅的点个数(包括$v$自身)
前者可以在LCT修改的过程中再维护一个树状数组(以优先级为下标,长度为权值),后者直接在LCT中查询该节点(将其Splay到根)即可,将两者求和即为答案
时间复杂度为$o(n\log n)$,可以通过
1 #include<bits/stdc++.h> 2 using namespace std; 3 #define N 200005 4 struct Edge{ 5 int nex,to; 6 }edge[N<<1]; 7 int E,n,m,x,y,head[N],f[N<<1],st[N],fa[N],val[N],sz[N],rev[N],ch[N][2]; 8 char s[11]; 9 int lowbit(int k){ 10 return (k&(-k)); 11 } 12 void update(int k,int x){ 13 while (k<=n+m){ 14 f[k]+=x; 15 k+=lowbit(k); 16 } 17 } 18 int query(int k){ 19 int ans=0; 20 while (k){ 21 ans+=f[k]; 22 k-=lowbit(k); 23 } 24 return ans; 25 } 26 int which(int k){ 27 return ch[fa[k]][1]==k; 28 } 29 int check(int k){//返回1当且仅当不是根节点 30 return ch[fa[k]][which(k)]==k; 31 } 32 void upd(int k){ 33 rev[k]^=1; 34 swap(ch[k][0],ch[k][1]); 35 } 36 void up(int k){ 37 sz[k]=sz[ch[k][0]]+sz[ch[k][1]]+1; 38 } 39 void down(int k){ 40 if (ch[k][0])val[ch[k][0]]=val[k]; 41 if (ch[k][1])val[ch[k][1]]=val[k]; 42 if (rev[k]){ 43 upd(ch[k][0]); 44 upd(ch[k][1]); 45 rev[k]=0; 46 } 47 } 48 void rotate(int k){ 49 int f=fa[k],g=fa[f],p=which(k); 50 fa[k]=g; 51 if (check(f))ch[g][which(f)]=k; 52 fa[ch[k][p^1]]=f,ch[f][p]=ch[k][p^1]; 53 fa[f]=k,ch[k][p^1]=f; 54 up(f),up(k); 55 } 56 void splay(int k){ 57 for(int i=k;;i=fa[i]){ 58 st[++st[0]]=i; 59 if (!check(i))break; 60 } 61 while (st[0])down(st[st[0]--]); 62 for(int i=fa[k];check(k);i=fa[k]){ 63 if (check(i)){ 64 if (which(k)==which(i))rotate(i); 65 else rotate(k); 66 } 67 rotate(k); 68 } 69 } 70 void access(int k){ 71 int lst=0; 72 while (k){ 73 splay(k); 74 update(val[k],sz[ch[k][1]]-sz[k]); 75 ch[k][1]=lst,up(k); 76 lst=k,k=fa[k]; 77 } 78 } 79 void make_root(int k){ 80 access(k); 81 splay(k); 82 upd(k); 83 } 84 void add(int x,int y){ 85 edge[E].nex=head[x]; 86 edge[E].to=y; 87 head[x]=E++; 88 } 89 void dfs(int k,int f){ 90 fa[k]=f,val[k]=k; 91 for(int i=head[k];i!=-1;i=edge[i].nex) 92 if (edge[i].to!=f){ 93 dfs(edge[i].to,k); 94 val[k]=max(val[k],val[edge[i].to]); 95 } 96 update(val[k],1); 97 if (val[k]==k){ 98 sz[k]=1; 99 return; 100 } 101 for(int i=head[k];i!=-1;i=edge[i].nex) 102 if ((edge[i].to!=f)&&(val[k]==val[edge[i].to])){ 103 sz[k]=sz[edge[i].to]+1; 104 ch[k][1]=edge[i].to; 105 return; 106 } 107 } 108 void Update(int k){ 109 make_root(k); 110 val[k]=++val[0]; 111 update(val[k],sz[k]); 112 } 113 int Query(int k){ 114 splay(k); 115 return query(val[k])-sz[ch[k][0]]; 116 } 117 int main(){ 118 scanf("%d%d",&n,&m); 119 memset(head,-1,sizeof(head)); 120 for(int i=1;i<n;i++){ 121 scanf("%d%d",&x,&y); 122 add(x,y),add(y,x); 123 } 124 dfs(n,0); 125 val[0]=n; 126 for(int i=1;i<=m;i++){ 127 scanf("%s%d",s,&x); 128 if (s[0]=='u')Update(x); 129 if (s[0]=='w')printf("%d\n",Query(x)); 130 if (s[0]=='c'){ 131 scanf("%d",&y); 132 if (Query(x)<Query(y))printf("%d\n",x); 133 else printf("%d\n",y); 134 } 135 } 136 return 0; 137 }View Code
这篇关于[cf1137F]Matches Are Not a Child's Pla的文章就介绍到这儿,希望我们推荐的文章对大家有所帮助,也希望大家多多支持为之网!
- 2025-01-09CMS内容管理系统是什么?如何选择适合你的平台?
- 2025-01-08CCPM如何缩短项目周期并降低风险?
- 2025-01-08Omnivore 替代品 Readeck 安装与使用教程
- 2025-01-07Cursor 收费太贵?3分钟教你接入超低价 DeepSeek-V3,代码质量逼近 Claude 3.5
- 2025-01-06PingCAP 连续两年入选 Gartner 云数据库管理系统魔力象限“荣誉提及”
- 2025-01-05Easysearch 可搜索快照功能,看这篇就够了
- 2025-01-04BOT+EPC模式在基础设施项目中的应用与优势
- 2025-01-03用LangChain构建会检索和搜索的智能聊天机器人指南
- 2025-01-03图像文字理解,OCR、大模型还是多模态模型?PalliGema2在QLoRA技术上的微调与应用
- 2025-01-03混合搜索:用LanceDB实现语义和关键词结合的搜索技术(应用于实际项目)