Codeforces 1140F 线段树 分治 并查集

Posted pkgunboat

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Codeforces 1140F 线段树 分治 并查集相关的知识,希望对你有一定的参考价值。

题意及思路:https://blog.csdn.net/u013534123/article/details/89010251

之前cf有一个和这个相似的题,不过那个题只有合并操作,没有删除操作,直接并查集搞一搞就行了。对于这个题,因为有删除操作,我们对操作序列建一颗线段树,记录每个操作影响的区间操作就可以了。这里的并查集不能路径压缩,要按秩合并,这样复杂度是O(logn)的。

代码:

#include <bits/stdc++.h>
#define ls (o << 1)
#define rs (o << 1 | 1)
#define INF 0x3f3f3f3f
#define db double
#define pii pair<int, int>
#define LL long long
using namespace std;
const int maxn = 300010;
const int Base = 300000;
vector<pii> tr[maxn * 4];
map<pii, int> mp;
map<pii, int>::iterator it;
LL res[maxn], cnt_x[maxn * 2], cnt_y[maxn * 2], sz[maxn * 2];
int f[maxn * 2];
LL ans;
pii a[maxn];
int get(int x) 
	if(x == f[x]) return x;
	return get(f[x]);

void add(int o, int l, int r, int ql, int qr, pii val) 
	if(l >= ql &&r <= qr) 
		tr[o].push_back(val);
		return;
	
	int mid = (l + r) >> 1;
	if(ql <= mid) add(ls, l, mid, ql, qr, val);
	if(qr > mid) add(rs, mid + 1, r, ql, qr, val);

void del(int x, int y) 
	int x1 = get(x), y1 = get(y);
	if(x1 != y1) return;
	ans -= cnt_x[x] * cnt_y[x];
	cnt_x[x] -= cnt_x[y], cnt_y[x] -= cnt_y[y];
	sz[x] -= sz[y];
	ans += cnt_x[x] * cnt_y[x];
	ans += cnt_x[y] * cnt_y[y];
	f[y] = y;

pii merge(int x, int y) 
	int x1 = get(x), y1 = get(y);
	if(x1 == y1) return make_pair(-1, -1);
	if(sz[x1] < sz[y1]) swap(x1, y1);
	ans -= cnt_x[x1] * cnt_y[x1];
	ans -= cnt_x[y1] * cnt_y[y1];
	sz[x1] += sz[y1];
	cnt_x[x1] += cnt_x[y1], cnt_y[x1] += cnt_y[y1];
	ans += cnt_x[x1] * cnt_y[x1];
	f[y1] = x1;
	return make_pair(x1, y1); 

void dfs(int o, int l, int r) 
	if(l == 12) 
		l++;
		l--;
	
	stack<pii> s;
	for (auto x : tr[o]) 
		pii tmp = merge(x.first, x.second);
		if(tmp.first != -1) s.push(tmp);
	
	if(l == r) res[l] = ans;
	else 
		int mid = (l + r) >> 1;
		dfs(ls, l, mid);
		dfs(rs, mid + 1, r);
	
	while(!s.empty()) 
		del(s.top().first, s.top().second);
		s.pop();
	

int main() 
	int n, x, y;
	scanf("%d", &n);
	for (int i = 1; i <= n; i++) 
		scanf("%d%d", &x, &y);
		y += Base;
		a[i] = make_pair(x, y);
		if(mp.find(a[i]) == mp.end()) mp[a[i]] = i;
		else 
			add(1, 1, n, mp[a[i]], i - 1, a[i]);
			mp.erase(a[i]);
		
	
	for (it = mp.begin(); it != mp.end(); it++) 
		add(1, 1, n, it -> second, n, it -> first);
	
	for (int i = 1; i <= Base; i++) 
		f[i] = i, cnt_x[i] = 1, cnt_y[i] = 0, sz[i] = 1;
	
	for (int i = Base + 1; i <= Base * 2; i++) 
		f[i] = i, cnt_x[i] = 0, cnt_y[i] = 1, sz[i] = 1;
	
	dfs(1, 1, n);
	for (int i = 1; i <= n; i++)
		printf("%lld ", res[i]);

  

以上是关于Codeforces 1140F 线段树 分治 并查集的主要内容,如果未能解决你的问题,请参考以下文章

Codeforces 938G 线段树分治 线性基 可撤销并查集

线段树分治

Codeforces 750E New Year and Old Subsequence 线段树 + dp (看题解)

线段树分治总结(线段树分治,线段树,并查集,树的dfn序,二分图染色)

CF576E Painting Edges [线段树分治,可撤销并查集]

CF576E Painting Edges [线段树分治,可撤销并查集]