经典算法 判断一个图是不是树

判断一个图是不是树

问题描述

给一个以0 0结尾的整数对列表,除0 0外的每两个整数表示一条连接了这两个节点的边。假设节点编号不超过100000大于0。你只要判断由这些节点和边构成的图是不是树。是输出YES,不是输出NO。

输入样例1

in 复制代码
6 8  5 3  5 2  6 4 5 6  0 0

输出样例1

in 复制代码
YES

输入样例2

in 复制代码
8 1  7 3  6 2  8 9  7 5 7 4  7 8  7 6  0 0

输出样例2

in 复制代码
YES

输入样例3

in 复制代码
3 8  6 8  6 4 5 3  5 6  5 2  0 0

输出样例3

in 复制代码
NO

输入样例4

in 复制代码
1 2 3 4 0 0

输出样例4

in 复制代码
NO

输入样例5

空树也是树

cpp 复制代码
0 0

输出样例5

in 复制代码
YES

c++代码

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

using namespace std;

int a, b, cont = 0;
int arr[100001];
bool key = true;
unordered_set<int> st;

int myfind(int x) {
    int root = x;
    while(root != arr[root]) root = arr[root];
    int i = x, j;
    while(i != root) {
        j = arr[i];
        arr[i] = root;
        i = j;
    }
    return root;
}

void mymerge(int x, int y) {
    x = myfind(x), y = myfind(y);
    if (x != y) arr[y] = arr[x];
}

int main() {
    for (int i = 1; i <= 100000; i++) arr[i] = i;
    while(true) {
        scanf("%d %d", &a, &b);
        if (a == 0 && b == 0) {
            if (cont == 0) {
                printf("YES\n");
                return 0;
            }
            if (key && st.size() != cont + 1) key = false;
            if (key) {
                int root = -1;
                for (int x : st) {
                    int k = myfind(x);
                    if (root == -1) root = k;
                    if (k != root) {
                        key = false;
                        break;
                    }
                }
            }
            if (key) printf("YES\n");
            else printf("NO\n");
            break;
        }
        else {
            cont++;
            if (st.find(a) == st.end()) st.insert(a);
            if (st.find(b) == st.end()) st.insert(b);
            if (key) {
                int x = myfind(a), y = myfind(b);
                if (x == y) key = false;
                else mymerge(a, b);
            }
        }
    }
    return 0;
}

算法解析

满足下面三个条件的图是树

1、不存在环。

2、所有点都是互相连通的。

3、点数=边数 + 1。

判断环

用并查集,给每个点一个初始的编号,并初始化所有节点的父亲为本身。每新加入一条边就把边相连的两个集合合并到一起,如果边相连的集合原本就是同一个,说明已经成环,不是生成树。

cpp 复制代码
int x = myfind(a), y = myfind(b);
if (x == y) key = false; //边相连的集合原本就是同一个,说明已经成环,不是生成树。
else mymerge(a, b); //否则合并一下

判断节点数和边数的关系

把点存在一个unordered_set里面就行,因为不能存重复的,边用一个cont就可以存

cpp 复制代码
cont++;
if (st.find(a) == st.end()) st.insert(a);
if (st.find(b) == st.end()) st.insert(b);
......
if (key && st.size() != cont + 1) key = false;

判断连通性

如果是连通的,根据我们之前的合并操作,每一个节点应该都属于同一个集合,如果不是,则不是树。

cpp 复制代码
int root = -1;
for (int x : st) {
    int k = myfind(x);
    if (root == -1) root = k;
    if (k != root) {
        key = false;
        break;
    }
}
相关推荐
橘颂TA9 分钟前
【剑斩OFFER】算法的暴力美学——除自身以外数组的乘积
算法·leetcode·职场和发展·结构与算法
源码之家10 分钟前
机器学习:基于python租房推荐系统 预测算法 协同过滤推荐算法 房源信息 可视化 机器学习-线性回归预测模型 Flask框架(源码+文档)✅
大数据·python·算法·机器学习·数据分析·线性回归·推荐算法
鑫—萍35 分钟前
C/C++精品算法——双指针(1)
c语言·c++·算法
铭哥的编程日记43 分钟前
深入浅出蓝桥杯:算法基础概念与实战应用(三)搜索
算法·蓝桥杯·深度优先
2501_941111891 小时前
低延迟系统C++优化
开发语言·c++·算法
2501_941111992 小时前
C++中的装饰器模式变体
开发语言·c++·算法
怕什么真理无穷2 小时前
C++_面试15_零拷贝
c++·面试·职场和发展
Espresso Macchiato2 小时前
Leetcode 3748. Count Stable Subarrays
算法·leetcode·职场和发展·leetcode hard·leetcode 3748·leetcode周赛476·区间求和
AA陈超2 小时前
ASC学习笔记0007:用于与GameplayAbilities系统交互的核心ActorComponent
c++·笔记·学习·ue5·虚幻引擎