图论---匈牙利算法求二分图最大匹配的实现

开始编程前分析设计思路和程序的整体的框架,以及作为数学问题的性质:

程序流程图:

数学原理:

求解二分图最大匹配问题的算法,寻找一个边的子集,使得每个左部点都与右部点相连,并且没有两条边共享相同的端点。使用深度优先搜索来寻找增广路径。当找到一条增广路径时,将路径上的边加入匹配集合中,并更新匹配状态。重复这个过程直到无法找到增广路径为止。

时间复杂度分析:

时间消耗来自于DFS搜索和增广路径的寻找。对于每个左侧未匹配的点u,需要进行一次DFS搜索。在最坏情况下,DFS搜索需要遍历所有的右侧点,因此时间复杂度为O(n)。由于需要进行n次DFS搜索,所以时间复杂度为O(n^2)。

源代码:

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

const int MAXN = 505; // 最大顶点数
int n, m, k; // n为左部点数,m为右部点数,k为总变数
vector<int> G[MAXN]; // 邻接表表示二分图
int match[MAXN]; // match[i]表示右边第i个点匹配的是左边的哪点,如果没有匹配则为-1
bool vis[MAXN]; // DFS中标记右边各点是否已经被访问过
//进行深度搜索最大匹配边
bool dfs(int u) {
    int i;
    for (i = 0; i < G[u].size(); i++) {
        int v = G[u][i];
        if (!vis[v]) {
            vis[v] = true;
            if (match[v] == -1 || dfs(match[v])) {
                match[v] = u;//对边进行存储
                return true;
            }
        }
    }
    return false;
}
//匈牙利算法
int hungarian() {
    int res = 0;
    memset(match, -1, sizeof(match));
    for (int i = 1; i <= n; i++) {
        memset(vis, false, sizeof(vis));
        if (dfs(i)) res++;
    }
    return res;
}
int main() {
    cin >> n >> m >> k; // 输入左部点数和右部点数
    int u, v;
    int i=0;
    for(;i<k;i++){
        cin >> u >> v ; // 输入边 (u, v),u属于左部,v属于右部
        G[u].push_back(v);
    }
    int max_matches = hungarian(); // 计算最大匹配数
    cout << "最大匹配边数:" << max_matches << endl;
    // 输出最大匹配数
    // 输出最大匹配的所有边
    cout << "最大匹配边为:" << endl;
    for (int v = 1; v <= m; v++) {
        if (match[v] != -1) { // 如果右部点v有匹配
            cout << "Edge: " << match[v] << " - " << v <<endl; // 输出匹配边
        }
    }
    system("pause");
    return 0;
}

测试用例:(两侧顶点数均大于10,且两侧顶点数不相等)

|------------------------------------------------------------------------------------------------------------------------------------------------------|
| |

输出结果:

相关推荐
似水এ᭄往昔4 小时前
【C++】--二叉搜索树
开发语言·数据结构·c++
水木姚姚4 小时前
C++程序创建(VS Code)
开发语言·c++
小龙报4 小时前
【算法通关指南:数据结构与算法篇(五)】树的 “自我介绍”:从递归定义到存储绝技(vector vs 链式前向星)
c语言·数据结构·c++·算法·链表·启发式算法·visual studio
666HZ6664 小时前
C语言——C++的引用
c语言·开发语言·c++
点云SLAM4 小时前
C++包装器之类型擦除(Type Erasure)包装器详解(4)
c++·算法·c++17·类型擦除·c++高级应用·c++包装器·函数包装
小尧嵌入式4 小时前
C++中的封装继承多态
开发语言·arm开发·c++
小年糕是糕手4 小时前
【C++同步练习】C++入门
开发语言·数据结构·c++·算法·pdf·github·排序算法
繁华似锦respect5 小时前
C++ 设计模式之工厂模式详细介绍
java·linux·c++·网络协议·设计模式
胖咕噜的稞达鸭5 小时前
算法入门:专题二分查找算法 模板总结 题目练手 :排序数组中查找元素的第一个和最后一个位置 第一个错误的版本 查找x的平方根 搜索插入位置 山脉数组的封顶索引
c语言·c++·算法·leetcode
未来之窗软件服务5 小时前
幽冥大陆(三十六)S18酒店门锁SDK rust语言——东方仙盟筑基期
开发语言·c++·rust·智能门锁·东方仙盟sdk·东方仙盟一体化