代码随想录算法营Day59 | 寻找存在的路径, 冗余连接,冗余连接II

寻找存在的路径

这题使用并查集即可。并查集加路径压缩。

cpp 复制代码
#include <iostream>
using namespace std;
int find(int* father,int u){
    return father[u] == u ? u : father[u] = find(father,father[u]);
}

bool isSame(int* father,int u,int v){
    return find(father,u) == find(father,v);
}

void join(int* father,int u,int v){
    u = find(father,u);
    v = find(father,v);
    if(u == v){
        return;
    }
    father[v] = u;
}

int main(){
    int N,M,source,destination,u,v;
    cin >> N >> M;
    int father[N+1];
    for(int i=1;i<=N;i++){
        father[i] = i;
    }
    for(int i=0;i<M;i++){
        cin >> u >> v;
        join(father,u,v);
    }
    cin >> source >> destination;
    if(isSame(father,source,destination)){
        cout<<1;
    }
    else{
        cout<<0;
    }
}

冗余连接

这题也是并查集问题,删除那个导致图存在环的边,我们可以通过判断新加入的边的两个点是否存在同一个father,如果他们是同一个father,那就说明这两个点是已经加入图中的点,如果在这两个点之间加一条边的话会导致环的产生。而如果是新加入图的点那他们的祖先应该是不同的。

cpp 复制代码
#include <iostream>
using namespace std;
int N;
int father[1001];

void init(){
    for(int i=1;i<=N;i++){
        father[i] = i;
    }
}

int find(int u){
    return father[u] == u ? u : father[u] = find(father[u]);
}

bool isSame(int u,int v){
    return find(u) == find(v);
}

void join(int u,int v){
    u = find(u);
    v = find(v);
    if(u == v) return;
    father[v] = u;
}
int main(){
    int u,v;
    cin >> N;
    init();
    for(int i=0;i<N;i++){
        cin >> u >> v;
        if(isSame(u,v)){
            printf("%d %d",u,v);
            break;
        }
        else{
            join(u,v);
        }
    }
    
}

冗余连接II

这道题与上题逻辑类似,不过他变成了有向图,这里需要判断删除的边是否能继续保证图变成有向树即除了根节点,其余的节点的入度都得是1且不存在环。所以我们需要收集那些入度为2的节点,并尝试去删除其中的边,使有向图变成一个有向树。

cpp 复制代码
#include <iostream>
#include <vector>
using namespace std;
int N;
int father[1001];
 
void init(){
    for(int i=1;i<=N;i++){
        father[i] = i;
    }
}
 
int find(int u){
    return father[u] == u ? u : father[u] = find(father[u]);
}
 
bool isSame(int u,int v){
    return find(u) == find(v);
}
 
void join(int u,int v){
    u = find(u);
    v = find(v);
    if(u == v) return;
    father[v] = u;
}

void getRemoveEdge(vector<vector<int>>& edges){
    init();
    for(vector<int>& v : edges){
        if(isSame(v[0],v[1])){
            printf("%d %d",v[0],v[1]);
            return;
        }
        else{
            join(v[0],v[1]);
        }
    }
}

bool isTreeAfterRemoveEdges(vector<vector<int>> &edges,int deleteEdge){
    init();
    for(int i=0;i<edges.size();i++){
        if(i==deleteEdge) continue;
        if(isSame(edges[i][0],edges[i][1])){
            return false;
        }
        join(edges[i][0],edges[i][1]);
    }
    return true;
}

int main(){
    int u,v;
    vector<vector<int>> edges;
    cin >> N;
    vector<int> inDegrees(N+1,0);
    for(int i=0;i<N;i++){
        cin >> u >> v;
        inDegrees[v]++;
        edges.push_back({u,v});
    }
    vector<int> vec;
    for(int i = N-1;i>=0;i--){
        if(inDegrees[edges[i][1]] == 2){
            vec.push_back(i);
        }
    }

    if(vec.size()>0){
        if(isTreeAfterRemoveEdges(edges,vec[0])){
            printf("%d %d",edges[vec[0]][0],edges[vec[0]][1]);
        }
        else{
            printf("%d %d",edges[vec[1]][0],edges[vec[1]][1]);
        }
        return 0;
    }

    getRemoveEdge(edges);
}
相关推荐
IT猿手1 小时前
2025最新群智能优化算法:海市蜃楼搜索优化(Mirage Search Optimization, MSO)算法求解23个经典函数测试集,MATLAB
开发语言·人工智能·算法·机器学习·matlab·机器人
IT猿手3 小时前
2025最新群智能优化算法:山羊优化算法(Goat Optimization Algorithm, GOA)求解23个经典函数测试集,MATLAB
人工智能·python·算法·数学建模·matlab·智能优化算法
Dream it possible!6 小时前
LeetCode 热题 100_字符串解码(71_394_中等_C++)(栈)
c++·算法·leetcode
My Li.7 小时前
c++的介绍
开发语言·c++
修己xj7 小时前
算法系列之深度优先搜索寻找妖怪和尚过河问题的所有方式
算法
开心比对错重要7 小时前
leetcode69.x 的平方根
数据结构·算法·leetcode
美狐美颜sdk8 小时前
什么是美颜SDK?从几何变换到深度学习驱动的美颜算法详解
人工智能·深度学习·算法·美颜sdk·第三方美颜sdk·视频美颜sdk·美颜api
m0_461502698 小时前
【贪心算法1】
算法·贪心算法
Doopny@8 小时前
数字组合(信息学奥赛一本通-1291)
数据结构·算法·动态规划
邪恶的贝利亚9 小时前
C++之序列容器(vector,list,dueqe)
开发语言·c++