zoukankan      html  css  js  c++  java
  • Codeforces Round #720 (Div. 2) D.Nastia Plays with a tree 树上最小链覆盖记录路径

    Codeforces Round #720 (Div. 2) D.Nastia Plays with a tree 树上最小链覆盖记录路径

    题意

    给出一颗树,定义bamboos为每个点度数不超过2的树,每次操作可以在树上删除一条边,添加一条边,问最少操作次数把原来的树变为一个bamboo

    [2 leq n leq 10^5 ]

    分析

    把加边删边分开来看,我们知道每删一条边就会多一个连通块,最终删了(x)次会形成(x + 1)个连通块,加了(x)条边相当于减少(x)个连通块

    于是原问题可以看成先删(x)条边,让它形成(x + 1)个bamboo,再逐一连接,(否则无论如何都没办法形成最终的bamboo)

    观察一下最终的bamboo,可以看成若干条互不相交的链,我们发现这就是经典的最小链覆盖问题,于是套用树上最小链覆盖的算法(贪心orDP,此处不展开赘述)即可得到(x)

    当然,此题还要求输出方案,那么只需要在dfs的时候多加几个数组来表示拐点和删边,这些就相对来说简单了

    主要是题做得少,不知道有这个算法

    代码

    #include<bits/stdc++.h>
    #define pii pair<int,int>
    #define fi first
    #define se second
    #define re register
    using namespace std;
    typedef long long ll;
    
    int rd(){
    	int x = 0;
    	char ch = getchar();
    	while(ch < '0' || ch > '9') {
    		ch = getchar();
    	}
    	while(ch >= '0' && ch <= '9') {
    		x = x * 10 + ch - '0';
    		ch = getchar();
    	}
    	return x;
    }
    
    vector<pair<pii,pii>> choose;
    const int maxn = 1e5 + 5;
    vector<pii> e[maxn];
    int siz[maxn],vis[maxn];
    int del[maxn];
    int ans;
    
    void dfs(int u,int fa){
    	int cnt = 0;
    	siz[u] = 1;
    	for(auto v:e[u]) {
    		if(v.fi == fa) continue;
    		dfs(v.fi,u);
    		siz[u] += siz[v.fi];
    		if(!vis[v.fi]) cnt++;
    		else {
    			ans++;
    			del[v.se] = 1;
    		}
    	}
    	if(cnt >= 2) {
    		siz[u] -= 2;
    		vis[u] = 1;
    		for(auto v:e[u]) {
    			if(v.fi == fa) continue;
    			if(cnt <= 2) break;
    			if(!vis[v.fi]) {
    				cnt--;
    				ans++;
    				del[v.se] = 1;
    			}
    		}
    	}
    	else if(cnt == 1) siz[u]--; 
    }
    
    vector<pii> bamboos;
    vector<int> leaves;
    int used[maxn];
    
    void dfs2(int u,int rt){
    	used[u] = 1;
    	int cnt = 0;
    	for(auto v:e[u]) {
    		if(used[v.fi] || del[v.se]) continue;
    		cnt++;
    		dfs2(v.fi,rt);
    	}
    	if(u == rt && cnt == 1) 
    	   leaves.push_back(u);
    	else if(!cnt) 
    		leaves.push_back(u);	
    }
    
    
    int main(){
    	int T = rd();
    	while(T--){
    		ans = 0;
    		int n = rd();
    		for(int i = 0;i <= n;i++) e[i].clear(),used[i] = vis[i] = siz[i] = del[i] = 0;
    		leaves.clear();
    		bamboos.clear();
    		choose.clear();
    		for(int i = 1;i < n;i++){
    			int x = rd();
    			int y = rd();
    			e[x].push_back(make_pair(y,i));
    			e[y].push_back(make_pair(x,i));
    		}
    		dfs(1,1);
    		for(int i = 1;i <= n;i++){
    			if(!used[i]) {
    				dfs2(i,i);
    				if(leaves.size() == 2) {
    					bamboos.push_back(make_pair(leaves[0],leaves[1]));
    				}
    				else bamboos.push_back(make_pair(leaves[0],leaves[0]));
    				leaves.clear();
    			}
    		}
    		vector<pii> Del,Add;
    		for(int i = 1;i <= n;i++){
    			for(auto v:e[i]) {
    				if(del[v.se]) {
    					if(i < v.fi)
    						Del.push_back(make_pair(i,v.fi));
    				}
    			}	
    		}
    		for(int i = 1;i < bamboos.size();i++){
    			Add.push_back(make_pair(bamboos[i - 1].se,bamboos[i].fi));
    		}
    		for(int i = 0;i < ans;i++)
    			choose.push_back(make_pair(Del[i],Add[i]));
    		cout << ans << '
    ';
    		for(auto it:choose){
    			cout << it.fi.fi << ' ' << it.fi.se << ' ' << it.se.fi << ' ' << it.se.se << '
    '; 
    		}
    	}
    }
    
  • 相关阅读:
    改善ERP的用户体验,个性化用户界面(Jquery 提供源码)
    一个美国富人在洗手间里陪女佣孩子吃晚餐的故事
    easyui datagrid 批量 提交 json 数据到服务器
    XML基础
    Asp.Net 上传大文件专题(1)概述:上传大文件的难点
    Asp.Net 上传大文件专题(3)从请求流中获取数据并保存为文件[下]
    Asp.Net上传大文件系列
    Asp.Net 上传大文件专题(3)从请求流中获取数据并保存为文件[上]
    XM操作类
    Asp.Net 上传大文件专题(4)利用ajax技术显示上传进度
  • 原文地址:https://www.cnblogs.com/hznumqf/p/14749052.html
Copyright © 2011-2022 走看看