给出一颗树,定义bamboos为每个点度数不超过2的树,每次操作可以在树上删除一条边,添加一条边,问最少操作次数把原来的树变为一个bamboo
把加边删边分开来看,我们知道每删一条边就会多一个连通块,最终删了\(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 << ‘\n‘;
for(auto it:choose){
cout << it.fi.fi << ‘ ‘ << it.fi.se << ‘ ‘ << it.se.fi << ‘ ‘ << it.se.se << ‘\n‘;
}
}
}
Codeforces Round #720 (Div. 2) D.Nastia Plays with a tree 树上最小链覆盖记录路径
原文:https://www.cnblogs.com/hznumqf/p/14749052.html