ICode9

精准搜索请尝试: 精确搜索
首页 > 其他分享> 文章详细

[ZJOI2016]大森林

2022-06-22 20:33:18  阅读:150  来源: 互联网

标签:lct qu ++ 虚点 ZJOI2016 c2 节点 森林


复盘 \(\text{zzq}\) 讲的题,来写篇题解祸害社会

可能是更好做的想法。

Description

给定 \(n\) 棵树和 \(m\) 次操作,其中每棵树均有一个初始节点(并设定为“生长节点”),操作格式有三种:

  1. 给定 \(l\) 和 \(r\) ,让区间 \([l,\ r]\) 内所有树的生长节点下添加一个子节点,子节点序号为上次 \(1\) 操作加的新点的序号 \(+1\) 。

  2. 给定 \(l\) ,\(r\) 和 \(x\) ,让区间 \([l,\ r]\) 的树的生长节点全部改到 \(x\) ,如果该树不存在 \(x\) 则不产生影响。

  3. 给定 \(x\) ,\(u\) 和 \(v\) ,查询第 \(x\) 棵树上 \(u\) 和 \(v\) 之间的距离,保证该树存在 \(u\) 和 \(v\) 。

\(n \leq 10 ^ 5,\ m \leq 2 \cdot 10 ^ 5\)

Analysis

啊啊你说老师这题我会用 \(\text{LCT}\) 维护建树,好家伙直接 \(O(n ^ 2 \log n)\) 就做完了。

那显然是不行的。

这玩意想要在先做是没什么可能的了,先离线。

看看 \(1\) 操作,乍一读感觉书上点的序号都是依序增加的。但是不对,序号为上次 $1$ 操作加的新点的序号 $+1$ ,什么意思,就是一个点 \(x\) 的出现只会在一个区间。然后一看操作都是区间修改,单点查询,噢,应该是可以用扫描线的。

更具体一点就是按照下标为第一关键字,时间戳为第二关键字,把所有操作排序。

差不多分析完了,准备实操。

Solution

当然前面说的暴力空间也是会炸飞,所以总的只能建一棵树,这就需要考虑相邻两棵树之间的差异。

差异主要就是两个区间操作带来的:

  1. 对于操作 \(1\) ,我们先前提到每种操作是有一个适用范围的,所以对适用范围去交之后加/删点??
  • 其实也完全不需要,反正查询保证了两点必然存在,多那么几个根本不存在的点也不会影响最终路径上点的数量,所以这个就嗯加就行了。
  1. 对于操作 \(2\) ,我们可能需要一个能够比较灵活的点能够把目标换来换去的那种,考虑建虚点。
  • 所以大概我们可以在区间开始的时候把虚点连到原先修改出生结点的地方,区间结束的时候在连回去。

这么一看,能支持连边删边还要统计树上距离的, \(\text{LCT}\) 是比较实用的了,只不过要把边权改成点权,本质上其实差不多。

所以这样就有了一个很直接的想法:

对于每次操作 \(2\) 都去建一个虚点,点权为零,然后把出生节点设为这个虚点,这样的话除了第一个虚点连的是初始节点,其他都连向的是上一个虚点。

所以如果按照时间戳来的话,只要每次加入一个新区间的时候把对应的虚点连到操作本身该连的节点上,脱离一个区间的时候连回去就行了。

但是因为边权改成了点权,所以对应的路径长度就需要考虑 \(lca\) 在原树上的 \(fa\) ,所以树的形态就不能轻易改变,我们就需要不换根 \(\text{LCT}\) ,不会,摆~

咋办嘞,规避不了考虑原树 \(lca\) 的话可以再构造呀,那是不是可以在开一个虚点呢??

因为要求虚点点权为零,所以说我们这样子:新建两个虚点,一个点权为 \(-1\) ,一个为 \(1\) ,然后 \(-1\) 连向上一个 \(1\) 的虚点,所有新加的实点连到 \(1\) ,每次断 \(-1\) 和上一个 \(1\) 的虚点连的边。

(好赖配个图嘛)

image

(没错就是长这样的)

这样子如果查询两个点在一个虚点下, \(lca\) 必然是虚点 \(1\) ,并且 \(lca\) 的 \(fa\) 点权是 \(-1\) ;假如不在一个虚点下,同样也会跑到一个虚点 \(1\) ,那这样路径和就定了,清楚了 lca 和 \(lca\) 的 \(fa\) ,那就不在乎树上父子关系了,硬上 \(\text{LCT}\) 就可以了。

(PS:因为每次新节点都会长到生长节点的下面,所有的生长节点除了初始节点全都被建成了虚点,所以无论在哪里 \(lca\) 始终会停在虚点(或者初始节点))

Code

Code

/*

*/
#include 
using namespace std;
typedef long long ll;
const int N = 6e5 + 10, M = 8e5 + 10;
int n, m, tot, c1, c2, L[N], R[N], ans[N];
struct mdzz {
	int p, t, op, x, y;
	bool operator < (const mdzz &it) const {
		return (p == it.p) ? (t == it.t ? op > it.op : t < it.t) : p < it.p;
	}
} qu[M];
struct ios {
	inline char gc() {
		static const int IN_LEN = 1 << 18 | 1;
		static char buf[IN_LEN], *s, *t;
		return (s == t) && (t = (s = buf) + fread(buf, 1, IN_LEN, stdin)), s == t ? -1 : *s++;
	}
	template  inline ios & operator >> (_Tp & x) {
		static char ch, sig; ch = gc(); sig = 0;
		for (; !isdigit(ch); ch = gc()) {if (ch == -1) return *this; sig |= (ch =='-');}
		for (x = 0; isdigit(ch); ch = gc()) x = (x << 3) + (x << 1) + (ch ^ 48);
		sig && (x = -x); return *this;
	}
} io;
inline double dead() {
	char ch = io.gc();
	ll s = 0, w = 1, k = 0;
	double m = 1; bool is = 0;
	while (!isdigit(ch)) {if (ch == '-') w = -1; ch = io.gc();}
	while (isdigit(ch) || ch == '.') {
		if (ch == '.') is = 1;
		else if (!is) s = (s << 3) + (s << 1) + (ch ^ 48);
		else k = (k << 3) + (k << 1) + (ch ^ 48), m *= 0.1;
		ch = io.gc();
	}
	return (m * k + s) * w;
}
inline string sead() {
	char ch = io.gc();
	string s;
	while (ch == ' ' || ch == '\n' || ch == '\t' || ch == '\r' || ch == EOF) ch = io.gc();
	while (ch != ' ' && ch != '\n' && ch != '\t' && ch != '\r' && ch != EOF) s += ch, ch = io.gc();
	return s;
}
struct LCT {
	#define ls(x) ch[x][0]
	#define rs(x) ch[x][1]
	int ch[N][2], fa[N], siz[N], rev[N], val[N], sta[N], top;
	inline bool pd(int x) {return rs(fa[x]) == x;}
	inline bool isrt(int x) {return ls(fa[x]) != x && rs(fa[x]) != x;}
	inline void pushup(int x) {siz[x] = siz[ls(x)] + val[x] + siz[rs(x)];}
	inline void pushdown(int x) {
		if (!rev[x]) return ;
		rev[ls(x)] ^= 1; rev[rs(x)] ^= 1; swap(ls(x), rs(x)); rev[x] = 0;
	}
	inline void rotate(int x) {
		int d = fa[x], t = pd(x);
		ch[d][t] = ch[x][t ^ 1]; fa[ch[x][t ^ 1]] = d;
		if (!isrt(d)) ch[fa[d]][pd(d)] = x;
		fa[x] = fa[d]; ch[x][t ^ 1] = d; fa[d] = x;
		pushup(d); pushup(x);
	}
	inline void update(int x) {
		for (; ; x = fa[x]) {sta[++top] = x; if (isrt(x)) break;}
		while (top) {pushdown(sta[top]); --top;}
	}
	inline void rotaup(int x) {rotate(pd(x) ^ pd(fa[x]) ? x : fa[x]); rotate(x);}
	inline void splay(int x) {
		update(x); for(; !isrt(fa[x]) && !isrt(x); rotaup(x));
		if (!isrt(x)) rotate(x);
	}
	inline int access(int x) {
		int y = 0;
		for (; x; y = x, x = fa[x]) {splay(x); rs(x) = y; pushup(x);}
		return y;
	}
	inline void makert(int x) {access(x); splay(x); rev[x] ^= 1;}
	inline void split(int x, int y) {makert(x); access(y); splay(y);}
	inline int findrt(int x) {
		access(x); splay(x);
		while (ls(x)) pushdown(x), x = ls(x);
		splay(x); return x;
	}
	inline void link(int x, int y) {makert(x); if (findrt(y) != x) fa[x] = y;}
	inline void cut(int x, int y) {
		if (findrt(x) != findrt(y)) return ;
		split(x, y);
		if(fa[x] != y || rs(x)) return ;
		fa[x] = ls(y) = 0; pushup(x);
	}
	inline int query(int x, int y) {
		int ans = 0, lca;
		access(x); splay(x); ans = siz[x];
		lca = access(y); splay(y); ans += siz[y];
		access(lca); splay(lca); ans -= (siz[lca] << 1);
		return ans;
	}
	#undef ls
	#undef rs
} lct;
int main() {
//	freopen("forest.in", "r", stdin);
//	freopen("forest.out", "w", stdout);
	io >> n >> m; c1 = 1; c2 = m;
	lct.val[1] = 1; L[1] = 1; R[1] = n;
	++c2; lct.val[c2] = -1, lct.link(c2, 1);
	++c2; lct.val[c2] = 1, lct.link(c2, c2 - 1);
	memset(ans, -1, sizeof(ans));
	for (int i = 1, op, l, r, x; i <= m; ++i) {
		io >> op;
		if (op == 2) {io >> x >> l >> r; qu[++tot] = (mdzz) {x, i, 2, l, r};}
		else if (!op) {
			++c1, lct.val[c1] = 1, lct.link(c1, c2); io >> L[c1] >> R[c1];
		}
		else {
			io >> l >> r >> x; l = max(l, L[x]); r = min(r, R[x]);
			if (l > r) continue;
			++c2; lct.val[c2] = -1, lct.link(c2, c2 - 1);
			++c2; lct.val[c2] = 1, lct.link(c2, c2 - 1);
			qu[++tot] = (mdzz) {l, i, 1, c2 - 1, c2 - 2};
			qu[++tot] = (mdzz) {l, i, 0, c2 - 1, x};
			qu[++tot] = (mdzz) {r + 1, i, 1, c2 - 1, x};
			qu[++tot] = (mdzz) {r + 1, i, 0, c2 - 1, c2 - 2};
		}
	}
	sort(qu + 1, qu + 1 + tot);
	for (int i = 1; i <= tot; ++i) {
		if (!qu[i].op) lct.link(qu[i].x, qu[i].y);
		else if (qu[i].op == 1) lct.cut(qu[i].x, qu[i].y);
		else ans[qu[i].t] = lct.query(qu[i].x, qu[i].y);
	}
	for (int i = 1; i <= m; ++i) if (~ans[i]) printf("%d\n", ans[i]);
	return 0;
}

标签:lct,qu,++,虚点,ZJOI2016,c2,节点,森林
来源: https://www.cnblogs.com/Illusory-dimes/p/16402170.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有