zoukankan      html  css  js  c++  java
  • Codeforces [Hello 2020] 1284F New Year and Social Network(图论匹配推理+lct)

    https://codeforces.com/contest/1284/problem/F

    题目大意:

    有两个大小为n的树T1和T2.

    T2中的每条边(u, v)可以匹配T1中u到v路径上的所有边。

    求最大匹配,并给出方案。

    (1<=n<=250000)

    题解:

    首先你需要观察样例大胆猜想一定有完美匹配。

    考虑T1中的一个叶子x和它的父亲y。

    显然的是,从T2中随便选一个端点是x的边,那个边所形成的路径一定会包含(x,y)。

    也就是说在T2中选一条以x为端点的边,并且删掉这条边之后,可以将剩余连向x的边连向y,这样是等价的,目标是使新的T2也是一棵树,这样可以归纳证明有解。

    显然需要选的边是x到y的路径上的第一条出边。

    这样得到了一个(O(n^2))的做法,考虑用lct暴力维护上面的操作,显然还是会被卡的。

    不需要把x相邻的边重连到y,只需要在删除选的那条边后,给x到y连一条边。

    这样可以实现到y的意思。

    但是这样以后就不太好维护一条边对应的原边是什么。

    所以把边也弄作点,原来的边权值是1,其它的(点或新加的边)的权值都是0,每次在x到y的路径上二分第一个权值为1的点即可。

    #include<bits/stdc++.h>
    #define fo(i, x, y) for(int i = x, _b = y; i <= _b; i ++)
    #define ff(i, x, y) for(int i = x, _b = y; i <  _b; i ++)
    #define fd(i, x, y) for(int i = x, _b = y; i >= _b; i --)
    #define ll long long
    #define pp printf
    #define hh pp("
    ")
    using namespace std;
    
    const int N = 750005;
    
    int n, x, y;
    vector<int> e1[N];
    int b[N][2];
    
    #define pb push_back
    #define si size()
    #define re resize
    
    int r[N], us[N], d[N], d0;
    
    namespace lct {
    	int t[N][2], fa[N], pf[N], rev[N];
    	int z[N], siz[N];
    	int lr(int x) { return t[fa[x]][1] == x;}
    	#define x0 t[x][0]
    	#define x1 t[x][1]
    	void upd(int x) {
    		siz[x] = siz[x0] + siz[x1] + z[x];
    	}
    	void ro(int x) {
    		int y = fa[x], k = lr(x);
    		t[y][k] = t[x][!k]; if(t[x][!k]) fa[t[x][!k]] = y;
    		fa[x] = fa[y]; if(fa[y]) t[fa[y]][lr(y)] = x;
    		t[x][!k] = y, fa[y] = x, pf[x] = pf[y];
    		upd(y); upd(x);
    	}
    	void fan(int x) {
    		swap(x0, x1), rev[x] ^= 1;
    	}
    	void down(int x) {
    		if(rev[x]) fan(x0), fan(x1), rev[x] = 0;
    	}
    	int d[N], d0;
    	void xc(int x) {
    		for(; x; x = fa[x]) d[++ d0] = x;
    		while(d0) down(d[d0 --]);
    	}
    	void sp(int x, int y) {
    		xc(x);
    		for(; fa[x] != y; ro(x)) if(fa[fa[x]] != y)
    			ro(lr(x) == lr(fa[x]) ? fa[x] : x);
    	}
    	void ac(int x) {
    		for(int y = 0; x; ) {
    			sp(x, 0), fa[x1] = 0, pf[x1] = x;
    			x1 = y, fa[y] = x, pf[y] = 0;
    			upd(x), y = x, x = pf[x];
    		}
    	}
    	void mr(int x) {
    		ac(x); sp(x, 0); fan(x);
    	}
    	void link(int x, int y) {
    		mr(x); pf[x] = y; ac(x);
    	}
    	void cut(int x, int y) {
    		mr(x); ac(y); sp(x, 0);
    		x1 = fa[y] = pf[y] = 0;
    	}
    	int fl(int x) {
    		down(x);
    		if(siz[x0]) return fl(x0);
    		return z[x] ? x : fl(x1);
    	}
    }
    
    int fa[N];
    
    void dg(int x) {
    	ff(j, 0, e1[x].si) if(fa[x] != e1[x][j])
    		fa[e1[x][j]] = x, dg(e1[x][j]);
    }
    
    int td;
    
    int main() {
    	scanf("%d", &n);
    	td = n;
    	fo(i, 1, n - 1) {
    		scanf("%d %d", &x, &y);
    		e1[x].pb(y); e1[y].pb(x);
    	}
    	fo(i, 1, n - 1) {
    		scanf("%d %d", &x, &y);
    		td ++;
    		lct :: z[td] = lct :: siz[td] = 1;
    		b[td][0] = x; b[td][1] = y;
    		lct :: link(x, td);
    		lct :: link(td, y);
    	}
    	dg(1);
    	fo(i, 1, n) r[fa[i]] ++;
    	fo(i, 1, n) if(!r[i]) d[++ d0] = i;
    	pp("%d
    ", n - 1);
    	fo(i, 1, n - 1) {
    		int x = d[i];
    		if(!(-- r[fa[x]])) d[++ d0] = fa[x];
    		lct :: mr(x);
    		lct :: ac(fa[x]);
    		lct :: sp(x, 0);
    		int z = lct :: fl(lct :: t[x][1]);
    		lct :: sp(z, 0);
    		
    		pp("%d %d %d %d
    ", x, fa[x], b[z][0], b[z][1]);
    		
    		lct :: cut(b[z][0], z);
    		lct :: cut(b[z][1], z);
    		
    		td ++;
    		lct :: link(x, td);
    		lct :: link(fa[x], td);
    	}
    }
    
    
  • 相关阅读:
    [算法分析]计数排序
    [置顶] 基于stm32f103zet6之UC/OS_II的学习1(初步移植OS点灯大法)
    IOS开发(59)之Block Object的调用
    【译】测试员,敢问路在何方?来自微软工程师
    各种字符串hash
    hdu 2579 BFS
    qq相册
    程序人生之我们的故事:十年如歌(9)
    关联模型和无限极分类
    十大技巧破解电话面试
  • 原文地址:https://www.cnblogs.com/coldchair/p/12162111.html
Copyright © 2011-2022 走看看