BZOJ 3319: 黑白树 并查集 + 离线 + 思维
Description
Input
Output
我们知道,并查集是向上合并的
如果正着做,每一次将一条路径染黑,会导致很多白点的祖先改变,而且是向下变
向下变就十分麻烦,非常不好做
不妨逆着操作,记录每一个节点变黑的最早时间,将操作逆着进行
可以先将整棵树全部染黑,逆着逐渐染白,那么每个白点的祖先只会更向上,向上合并
比如当前要将点 $x$ 染白,那么 $x$ 子树中的白点在改动前并查集指向的祖先都是 $x$
在改动后 $x$ 的祖先会向上合并,而 $x$ 子树中的白点在并查集查祖先时查到 $x$ 的话会顺着 $x$ 向上合并到的祖先继续向上查询
- #include<cstdio>
- #include<algorithm>
- #include<cstring>
- #include<map>
- #include<vector>
- #include<stack>
- #include<queue>
- #define setIO(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)
- #define maxn 1002002
- using namespace std;
- char *p1,*p2,buf[100000];
- #define nc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,100000,stdin),p1==p2)?EOF:*p1++)
- int rd() {int x=0; char c=nc(); while(c<48) c=nc(); while(c>47) x=(((x<<2)+x)<<1)+(c^48),c=nc(); return x;}
- vector<int>G[maxn];
- stack<int>S;
- int n,m,edges;
- int tm[maxn],hd[maxn<<1],to[maxn<<1],nex[maxn<<1];
- int siz[maxn],top[maxn],fa[maxn],dep[maxn],son[maxn];
- int idx[maxn<<1],id[maxn<<1];
- inline void addedge(int u,int v,int c) {
- nex[++edges]=hd[u],hd[u]=edges,to[edges]=v, id[edges]=c;
- }
- void dfs1(int u,int ff) {
- dep[u]=dep[ff]+1,siz[u]=1,fa[u]=ff;
- for(int i=hd[u];i;i=nex[i]) {
- int v=to[i];
- if(v==ff) continue;
- idx[v]=id[i], dfs1(v,u), siz[u]+=siz[v];
- if(siz[v] > siz[son[u]]) son[u] = v;
- }
- }
- void dfs2(int u,int tp) {
- top[u]=tp;
- if(son[u]) dfs2(son[u], tp);
- for(int i=hd[u];i;i=nex[i]) {
- int v=to[i];
- if(v==fa[u] || v==son[u]) continue;
- dfs2(v,v);
- }
- }
- inline int LCA(int x,int y) {
- while(top[x]^top[y]) {
- dep[top[x]]>dep[top[y]]?x=fa[top[x]]:y=fa[top[y]];
- }
- return dep[x]<dep[y]?x:y;
- }
- struct Opt {
- int opt,x,y;
- }op[maxn];
- struct Union {
- int p[maxn];
- inline void init() {
- for(int i=0;i<maxn;++i) p[i]=i;
- }
- int find(int x) {
- return p[x]==x?x:p[x]=find(p[x]);
- }
- }black,white;
- inline void mark(int u,int lca,int cur) {
- u=tm[u]?black.find(u):u;
- while(dep[u]>dep[lca]) {
- tm[u]=cur;
- if(!tm[fa[u]]) {
- black.p[u]=fa[u];
- u=fa[u];
- }
- else {
- int y=black.find(fa[u]);
- black.p[u]=y;
- u=y;
- }
- }
- }
- inline void update(int u,int v,int cur) {
- int lca=LCA(u,v);
- mark(u,lca,cur), mark(v,lca,cur);
- }
- inline void change(int u) {
- white.p[u]=white.find(fa[u]);
- }
- int main() {
- // setIO("input");
- scanf("%d%d",&n,&m);
- for(int i=1;i<n;++i) {
- int u,v;
- u=rd(),v=rd();
- addedge(u,v,i),addedge(v,u,i);
- }
- dfs1(1,0), dfs2(1,1), black.init();
- for(int i=1;i<=m;++i) {
- op[i].opt=rd();
- if(op[i].opt==1) op[i].x=rd();
- if(op[i].opt==2) op[i].x=rd(), op[i].y=rd(), update(op[i].x,op[i].y,i);
- }
- white.init();
- for(int i=2;i<=n;++i) G[tm[i]==0?m+1:tm[i]].push_back(i);
- m+=(G[m+1].size()>1);
- for(int i=m;i>=1;--i) {
- if(G[i].size()) {
- for(int j=0;j<G[i].size();++j) change(G[i][j]);
- }else if(op[i].opt==1) {
- S.push(white.find(op[i].x));
- }
- }
- while(!S.empty()) {
- printf("%d\n",idx[S.top()]); S.pop();
- }
- return 0;
- }
BZOJ 3319: 黑白树 并查集 + 离线 + 思维的更多相关文章
- BZOJ 3319 黑白树 并查集+线段树
这这这这这这什么毒瘤题!!!!!!!!!!!!!!!!!!!!!!!!!!!! 卡LCT(优秀的LCT由于是均摊本身就带着2,3的常数在,而且这道题对于LCT标记十分难维护,又得乘上4,5然后就炸了) ...
- poj 2528 Mayor's posters 线段树 || 并查集 离线处理
题目链接 题意 用不同颜色的线段覆盖数轴,问最终数轴上有多少种颜色? 注:只有最上面的线段能够被看到:即,如果有一条线段被其他的线段给完全覆盖住,则这个颜色是看不到的. 法一:线段树 按题意按顺序模拟 ...
- [BZOJ 3319] 黑白树
3319: 黑白树 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 557 Solved: 194[Submit][Status][Discuss] ...
- BZOJ 3319: 黑白树 树+并查集+未调完+神题
Code: #include<bits/stdc++.h> #define maxn 1000003 using namespace std; char *p1,*p2,buf[10000 ...
- 【BZOJ3319】黑白树 并查集
[BZOJ3319]黑白树 Description 给定一棵树,边的颜色为黑或白,初始时全部为白色.维护两个操作:1.查询u到根路径上的第一条黑色边的标号.2.将u到v 路径上的所有边的颜色设为 ...
- BZOJ 3211 线段树+并查集
思路: 我们很容易发现 一个数开根号 开几(很小)次 就到了1 1 再怎么开 都是1 由于这个性质 我们就可以用并查集 了 //By SiriusRen #include <cmath> ...
- [HDU3710] Battle Over Cities [树链剖分+线段树+并查集+kruskal+思维]
题面 一句话题意: 给定一张 N 个点, M 条边的无向连通图, 每条边上有边权 w . 求删去任意一个点后的最小生成树的边权之和. 思路 首先肯定要$kruskal$一下 考虑$MST$里面去掉一个 ...
- 【BZOJ】3319: 黑白树(并查集+特殊的技巧/-树链剖分+线段树)
http://www.lydsy.com/JudgeOnline/problem.php?id=3319 以为是模板题就复习了下hld............................. 然后n ...
- 【BZOJ】3319: 黑白树
http://www.lydsy.com/JudgeOnline/problem.php?id=3319 题意:给一棵n节点的树(n<=1e6),m个操作(m<=1e6),每次操作有两种: ...
随机推荐
- Servlet 三种创建方式
servlet 是运行在 Web 服务器(tomcat)中的小型 Java 程序(即:服务器端的小应用程序) (其实就是一个java类,只不过不用再new了).servlet 通常通过 HTTP(超文 ...
- IP子网的划分
一.划分子网的具体步骤 已知192.168.1.0/24,划分8个子网(一个ip划分8个子网,即网络位不再是24位)1100 0000.1010 1000.0000 0001.0000 00001 . ...
- 安装Git并关联
下载git 打开git bash 生成key将 key绑定到帐号 输入命令 ssh-keygen -t rsa -C 'LoginName' 根据命令生成的地址找到对应文件复制密钥 打开github登 ...
- Java String == && equal
[.net超级群:27921837] Java中equals和==的区别 java中的数据类型,可分为两类:1.基本数据类型,也称原始数据类型.byte,short,char,int,long,flo ...
- linux sed 命令 实现对文件的增删改替换查 实验
1. 统一实验文本 # 创建包含下面内容的文件,后面的操作都会使用这个文件 [root@MongoDB ~]# cat person.txt ,mike,CEO ,jack,CTO ,yy,CFO , ...
- Nginx跨域问题
Nginx跨域无法访问,通常报错: Failed to load http://172.18.6.30:8086/CityServlet: No 'Access-Control-Allow-Origi ...
- JS跨域--window.name
JS跨域--window.name:https://www.jianshu.com/p/43ff69d076e3
- phpstudy添加PHP
想在phpstudy2018里面增加一个php版本,操作如下: 一.下载php-7.2.19-ts文件,解压缩,放在相应的目录下: 二.修改Apache的配置文件1.修改httpd.conf 配置,D ...
- Java关于继承中的内存分配
1.定义 super:当前对象的父类对象 this :当前对象,谁调用this所在的方法,this就是哪一个对象. 2.内存分析 另一个例子: public s ...
- Houdini:也许是你未曾听过的最振奋人心的 CSS 进化
原文链接:Houdini: Maybe The Most Exciting Development In CSS You’ve Never Heard Of更多译文将陆续推出,欢迎点赞+收藏+关注我的 ...