代码随想录算法训练营Day-55 图论06 | 108.冗余连接、109.冗余连接II

108.冗余连接

并查集判断冗余边:边的两个端点已经连通,则这条边就是成环边;题目要最后出现的成环边,所以遍历完所有边,输出最后记录的一条。

cpp 复制代码
#include<iostream>
#include<vector>
using namespace std;

int n=1005;
vector<int> father = vector<int>(n,0);

void _init(){
    for(int i=0;i<n;i++){
        father[i] = i;
    }
}

int find(int a){
    if(father[a] == a) return a;
    else return father[a] = find(father[a]);
}

void join(int a,int b){
    a = find(a);
    b = find(b);
    if(a==b) return;
    else father[a] = b;
}

bool isSame(int a,int b){
    a = find(a);
    b = find(b);
    return a==b;
}

int main(){
    int s,t;
    cin>>n;
    _init();
    int ans_s=0,ans_t=0;
    for(int i=0;i<n;i++){
        cin>>s>>t;
        if(isSame(s,t)){
            ans_s = s;
            ans_t = t;
        }else join(s,t);
    }
    cout<<ans_s<<" "<<ans_t<<endl;
}

109.冗余连接II

两种情况:

1.存在入度为2的节点-找到连到该节点的两条边,优先删除后出现的边,如果删除后无环,就直接输出该边,否则删除前面那条。

2.不存在入度为2的节点,说明有环,退化成上一题并查集寻找冗余边。

cpp 复制代码
#include<iostream>
#include<vector>
using namespace std;

int n=1005;
vector<int> father = vector<int>(n,0);

void _init(){
    for(int i=0;i<n;i++){
        father[i] = i;
    }
}

int find(int a){
    if(father[a] == a) return a;
    else return father[a] = find(father[a]);
}

void join(int a,int b){
    a = find(a);
    b = find(b);
    if(a==b) return;
    else father[a] = b;
}

bool isSame(int a,int b){
    a = find(a);
    b = find(b);
    return a==b;
}

bool isTreeAfterRemoved(vector<vector<int>>& edges, int remove){
    for(int i=0;i<edges.size();i++){
        if(i == remove) continue;

        if(isSame(edges[i][0],edges[i][1])) return false;
        else join(edges[i][0],edges[i][1]);
    }
    return true;
}

int main(){
    _init();
    cin>>n;
    
    vector<vector<int>> edges(n,vector<int>(2));
    vector<int> indegree(n+1, 0);

    //记录所有有向边及顺序,并统计每个节点的入度
    for(int i=0;i<n;i++){
        cin>>edges[i][0]>>edges[i][1];
        indegree[edges[i][1]]++;
    }

    int twoIndegreedot=0;
    for(int i=1;i<=n;i++){
        if(indegree[i] == 2) twoIndegreedot = i;
    }
    if(twoIndegreedot!=0){
        int edge1 = -1, edge2 = -1;
        for(int i=0;i<edges.size();i++){
            if(edges[i][1] == twoIndegreedot){
                if(edge1==-1) edge1 = i;
                else edge2 = i;
            }
        }
        if(isTreeAfterRemoved(edges, edge2)){
            cout<<edges[edge2][0]<<" "<<edges[edge2][1];
            return 0;
        }else{
            cout<<edges[edge1][0]<<" "<<edges[edge1][1];
            return 0;
        }
    }else{//否则就只是去掉最后一个成环边就行
        int ans_1=0,ans_2=0;
        for(int i=0;i<edges.size();i++){
            if(isSame(edges[i][0],edges[i][1])){
                ans_1 = edges[i][0];
                ans_2 = edges[i][1];
            }else join(edges[i][0],edges[i][1]);
        }
        cout<<ans_1<<" "<<ans_2<<endl;
        return 0;
    }


}

相关推荐
fqbqrr4 小时前
2606C++,C++构的多态
开发语言·c++
小欣加油5 小时前
leetcode56 合并区间
c++·算法·leetcode·职场和发展
lqqjuly5 小时前
前沿算法深度解析(二)
人工智能·算法·机器学习
Yolo_TvT6 小时前
C++:析构函数
c++
徐小夕7 小时前
万字长文!千万级文档 RAG 知识库系统落地实践
前端·算法·github
akunkuntaimei7 小时前
2026年高考数学各省真题及答案(完整版)
算法·高考
Hello:CodeWorld7 小时前
C 风格变参 vs C++ 变参模板:核心区别与选型指南
c语言·c++·算法
8Qi89 小时前
LeetCode 516:最长回文子序列
算法·leetcode·职场和发展·动态规划
搬砖魁首10 小时前
基础能力系列 - 多线程2 - 条件变量
c++·rust·条件变量·原子类型·线程同步互斥
youngerwang10 小时前
【从搬运工到协处理器:网卡芯片架构、算法、验证与边缘演进深度剖析】
网络·算法·架构·芯片