代码随想录算法训练营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;
    }


}

相关推荐
BothSavage10 小时前
Trae远程开发中DeepSeek自定义模型4054错误的排查与修复
算法
小林ixn10 小时前
从暴力到KMP:一道题彻底搞懂字符串匹配的前世今生
算法
烬羽11 小时前
字符串算法入门:从反转字符串到回文判断,面试不再慌
算法·面试
郝学胜_神的一滴12 小时前
CMake 034:生成器表达式:解耦构建时序、精简分支逻辑的终极利器
c++·cmake
先吃饱再说1 天前
判断回文字符串,从一行代码到双指针优化
算法
见过夏天1 天前
C++ 基础入门完全指南
c++
黄敬峰1 天前
深入理解算法核心:从递归思想、数组扁平化到快速排序
算法
得物技术1 天前
从狂野代码到按目标生产:得物推荐 AI Harness 的工程化实践|AICon 演讲整理
人工智能·算法·架构
AI小老六1 天前
SkillOpt 架构拆解:把 Skill 文本当参数,用执行轨迹训练 Agent
后端·算法·ai编程