树上行走

题目描述

牛牛苦练武功绝学——轻功水上漂,最终没有练成,但是他学会了在树上行走的本领。

这天,牛牛落入了敌人的陷阱,身后有巨石追击,面前有n个点,n-1条边连成一张连通图(一棵树),现在牛牛必须立马选择进入这张图中,但是牛牛发现,这张图有两种不同的点,一旦进入一个点,所有与该点不同类型的点都会消失(相连的边也会消失),牛牛只能走到有边相连的点,牛牛想要自己尽量有更多的点可以活动,那么他可以进入哪些点?

题目分析:因为在补其他题目的时候不小心看到这个题目考察的是并查集,但是不论我怎么想都不知道该怎么用并查集去解决这个问题,思考无果之后,我看了代码,代码带给我的震撼很大,并查集是我学习的第一个算法,但是这种解题思路却是我第一次见,这种思想,代码让人耳目一新,真的太爽了!!代码非常巧妙的运用并查集的思想解决了问题。

方法一:并查集

#include <bits/stdc++.h>
using namespace std;
const int maxn = 1e6 + 10;
int a[maxn], f[maxn], b[maxn];

int find(int x) //寻找祖先
{
    if (x != f[x])
        f[x] = find(f[x]);
    return f[x];
}
int main()
{
    int n;
    scanf("%d", &n);
    for (int i = 1; i <= n; ++i)
    {
        scanf("%d", &a[i]);
        f[i] = i; //初始化元素
    }
    for (int i = 1; i < n; ++i)
    {
        int u, v;
        scanf("%d%d", &u, &v);
        if (a[u] == a[v]) //把相同类型的元素合并起来
            f[find(u)] = find(v);
    }
    vector<int> v;
    vector<int>::iterator it;
    int ans = 0;
    for (int i = 1; i <= n; ++i)
    {
        b[find(i)]++; //只对祖先元素进行加操作,同意一下联通个数
        ans = max(ans, b[find(i)]);
    }
    for (int i = 1; i <= n; ++i)
    {
        if (b[find(i)] == ans) //如果某个元素与祖先相通那么就记录一下
            v.push_back(i);
    }
    printf("%d\n", v.size());
    for (it = v.begin(); it != v.end(); it++)
        printf("%d ", *it);
}

方法二:dfs+前向星存图

#include <bits/stdc++.h>
using namespace std;
const int maxn = 1e6 + 10;
int cnt, tot, pre[maxn], col[maxn], head[maxn], num, vis[maxn], ans;
struct node
{
    int to, nex;
} w[maxn];

void add(int a, int b) //前向星存边
{
    w[++cnt].to = b;
    w[cnt].nex = head[a];
    head[a] = cnt;
}

void dfs(int now)
{
    ++num;
    vis[now] = tot;                          //标记当前点已经走过,同时初始化
    for (int i = head[now]; i; i = w[i].nex) //遍历当前点的邻接点
    {
        int to = w[i].to;                    //记录当前点的下一个点
        if (!vis[to] && col[now] == col[to]) //如果当前点的下一个点没有被访问,并且两个点的类型相同
            dfs(to);                         //就遍历下一个点
    }
}
int main()
{
    int n;
    scanf("%d", &n);
    for (int i = 1; i <= n; ++i)
        scanf("%d", &col[i]);
    for (int i = 1; i < n; ++i)
    {
        int a, b;
        scanf("%d %d", &a, &b);
        add(a, b);
        add(b, a);
    }
    for (int i = 1; i <= n; ++i)
    {
        if (!vis[i]) //如果当前点没有被访问
        {
            ++tot, num = 0;
            dfs(i);         //递归遍历
            col[tot] = num; //把个点所能走到的最大的点的数量赋给数组
            ans = max(ans, num);
        }
    }
    int g = 0;
    for (int i = 1; i <= n; ++i)
        if (col[vis[i]] == ans) //如果当前点能到的点的数量等于最大值,那么说明当前点属于最大值的一部分
            pre[++g] = i;
    sort(pre + 1, pre + g + 1);
    printf("%d\n", g);
    for (int i = 1; i <= g; ++i)
        printf("%d%c", pre[i], i == g ? '\n' : ' ');
}