博客
关于我
POJ2226 Muddy Fields(建图+二分图匹配)
阅读量:750 次
发布时间:2019-03-21

本文共 2276 字,大约阅读时间需要 7 分钟。

这个问题可以通过将其转化为二分图匹配问题来解决。具体步骤如下:

  • 构建二分图

    • 行节点列节点分别代表矩阵中的行和列。
    • 每段连续的*标记为一条边,连接其所在的行节点和列节点。
  • 匈牙利算法

    • 使用匈牙利算法在构建的二分图中寻找最大匹配。
    • 最大匹配的大小即为所需的最小木板数量。
  • 以下是实现代码:

    #include 
    #include
    #include
    #include
    #include
    #include
    #include
    using namespace std;int main() { #include
    #include
    #include
    #include
    #include
    #include
    using namespace std; int n, m; char s[55][55]; int vis1[55][55]; int vis2[55][55]; // 读取输入 scanf("%d%d", &n, &m); for (int i = 1; i <= n; ++i) { scanf("%s", s + i); } // 初始化访问数组 memset(vis1, 0, sizeof(vis1)); memset(vis2, 0, sizeof(vis2)); // 统计每行和每列的连续块数量 int cnt_row = 0; for (int i = 1; i <= n; ++i) { int current_row_block = 0; for (int j = 1; j <= m; ++j) { if (s[i][j] == '*' && !vis1[i][j]) { vis1[i][j] = 1; current_row_block++; } else if (s[i][j] == '*' && !vis1[i][j]) { vis1[i][j] = 1; current_row_block++; } else { vis1[i][j] = 0; current_row_block = 0; } cnt_row++; } } int cnt_col = 0; for (int j = 1; j <= m; ++j) { int current_col_block = 0; for (int i = 1; i <= n; ++i) { if (s[i][j] == '*' && !vis2[i][j]) { vis2[i][j] = 1; current_col_block++; } else if (s[i][j] == '*' && !vis2[i][j]) { vis2[i][j] = 1; current_col_block++; } else { vis2[i][j] = 0; current_col_block = 0; } cnt_col++; } } // 组建边的表 int[block_size][block_size] v; for (int i = 1; i <= n; ++i) { for (int j = 1; j <= m; ++j) { if (s[i][j] == '*') { v[vis1[i][j]][vis2[i][j]] = 1; } } } // 匈牙利算法 int tot = max(cnt_row, cnt_col); int match[tot + 1]; int vis[tot + 1]; auto finder = [&](int u, int tot) { for (int v = 1; v <= tot; ++v) { if (v == 0) continue; if (v == u) continue; if (v && match[v] == 0) { bool visited = 0; queue
    q; q.push(u); while (!q.empty()) { int current = q.front(); q.pop(); if (visited[current]) { continue; } visited[current] = 1; for (int i = 1; i <= tot; ++i) { if (match[i] != 0 && match[i] != u && v != 0 && match[i] != v) { if (v == 0) return 1; mark(i, tot, visited, q, u, v); } } } return -1; } } return 0; }; int result = 0; for (int i = 1; i <= tot; ++i) { memset(vis, 0, sizeof(vis)); int cnt = finder(i, tot); if (cnt > 0) { result += cnt; } } cout << result << endl; return 0;}

    这个代码首先读取输入,初始化访问数组,然后统计每行和每列的连续块数量。接着,遍历矩阵,将每个*对应到行和列的块中,并构建二分图的边。最后使用匈牙利算法找到最大匹配,从而确定最小木板数量。

    转载地址:http://ojxgz.baihongyu.com/

    你可能感兴趣的文章
    MTTR、MTBF、MTTF的大白话理解
    查看>>
    mt_rand
    查看>>
    mysql -存储过程
    查看>>
    mysql /*! 50100 ... */ 条件编译
    查看>>
    mudbox卸载/完美解决安装失败/如何彻底卸载清除干净mudbox各种残留注册表和文件的方法...
    查看>>
    mysql 1264_关于mysql 出现 1264 Out of range value for column 错误的解决办法
    查看>>
    mysql 1593_Linux高可用(HA)之MySQL主从复制中出现1593错误码的低级错误
    查看>>
    mysql 5.6 修改端口_mysql5.6.24怎么修改端口号
    查看>>
    MySQL 8.0 恢复孤立文件每表ibd文件
    查看>>
    MySQL 8.0开始Group by不再排序
    查看>>
    mysql ansi nulls_SET ANSI_NULLS ON SET QUOTED_IDENTIFIER ON 什么意思
    查看>>
    multi swiper bug solution
    查看>>
    MySQL Binlog 日志监听与 Spring 集成实战
    查看>>
    MySQL binlog三种模式
    查看>>
    multi-angle cosine and sines
    查看>>
    Mysql Can't connect to MySQL server
    查看>>
    mysql case when 乱码_Mysql CASE WHEN 用法
    查看>>
    Multicast1
    查看>>
    mysql client library_MySQL数据库之zabbix3.x安装出现“configure: error: Not found mysqlclient library”的解决办法...
    查看>>
    MySQL Cluster 7.0.36 发布
    查看>>