2022 icpc 西安站 L. Tree - 拓扑排序

题面

分析

题意就是求集合个数,满足集合所有点都是子节点和父节点关系或者不存在祖孙关系。那么可以将树拆分成若干条链,然后每次减少链数,将减少的链转化成另一种情况,也就是枚举所有链数的方案,取最小值。

对于求链数,也就是统计叶节点的个数,可以通过拓扑排序进行统计叶节点数量,然后通过删去一层叶节点来减少链数,实现了枚举所有方案。

代码
cpp 复制代码
#include <bits/stdc++.h>

#define int long long

using namespace std;

const int N = 1e6 + 10;

int fa[N];
int d[N];

void solve() {
    int n;
    cin >> n;
    for(int i = 1; i <= n; i ++) {
        fa[i] = 0;
        d[i] = 0;
    }
    for(int i = 2; i <= n; i ++) {
        int x;
        cin >> x;
        fa[i] = x;
        d[i] ++;
        d[x] ++;
    }
    queue<int> q;
    for(int i = 2; i <= n; i ++) {
        if(d[i] == 1) {
            q.push(i);
        }
        d[i] --;
    }
    int ans = n;
    int k = 0;
    while(q.size()) {
        int sz = q.size();
        ans = min(ans, k + sz);
        k ++;
        while(sz --) {
            int t = q.front();
            q.pop();
            if(fa[t] == 0) continue;
            d[fa[t]] --;
            if(d[fa[t]] == 0) q.push(fa[t]);
        }
    }
    cout << ans << "\n";
}

signed main() {
    ios::sync_with_stdio(false);
    cin.tie(nullptr);

    int T;
    cin >> T;
    while(T --) {
        solve();
    }
}
相关推荐
草履虫建模3 小时前
力扣算法 1768. 交替合并字符串
java·开发语言·算法·leetcode·职场和发展·idea·基础
naruto_lnq5 小时前
分布式系统安全通信
开发语言·c++·算法
Jasmine_llq5 小时前
《P3157 [CQOI2011] 动态逆序对》
算法·cdq 分治·动态问题静态化+双向偏序统计·树状数组(高效统计元素大小关系·排序算法(预处理偏序和时间戳)·前缀和(合并单个贡献为总逆序对·动态问题静态化
爱吃rabbit的mq6 小时前
第09章:随机森林:集成学习的威力
算法·随机森林·集成学习
(❁´◡`❁)Jimmy(❁´◡`❁)6 小时前
Exgcd 学习笔记
笔记·学习·算法
YYuCChi7 小时前
代码随想录算法训练营第三十七天 | 52.携带研究材料(卡码网)、518.零钱兑换||、377.组合总和IV、57.爬楼梯(卡码网)
算法·动态规划
CSDN_RTKLIB7 小时前
【四个场景测试】源文件编码UTF-8 BOM
c++
不能隔夜的咖喱7 小时前
牛客网刷题(2)
java·开发语言·算法
VT.馒头7 小时前
【力扣】2721. 并行执行异步函数
前端·javascript·算法·leetcode·typescript
进击的小头7 小时前
实战案例:51单片机低功耗场景下的简易滤波实现
c语言·单片机·算法·51单片机