G - Oil Skimming - hdu 4185(二分图匹配)

题意:在大海里有一些石油 ‘#’表示石油, ‘.’表示水,有个人有一个工具可以回收这些石油,不过只能回收1*2大小的石油块,里面不能含有海水,要不就没办法使用了,求出来最多能回收多少块石油

分析:先把数据处理一下,给每一点石油都进行编号,然后查找一下四周联合是否能组成石油块,能的话就连接,因为一点有可能即在左边又在右边,所以最后的结果应该除去 2

*************************************************************************

#include<stdio.h>
#include<string.h>
#include<queue>
using namespace std;

const int MAXN = 40005;///处理后点的个数
const int MAXM = 605;///原图大小
const int oo = 1e9+7;

char G[MAXM][MAXM]; int N;///保存原图
int  Index[MAXM][MAXM];///给石油点编号

struct Edge{int v, next;}e[MAXN*4];
int Head[MAXN], cnt;///处理后的边

int Mx[MAXN], My[MAXN];///记录与之匹配的点
int used[MAXN], dx[MAXN], dy[MAXN];///dx,dy记录BFS后的层次
int depth, NX;///宽搜的深度,和点的个数

void InIt()
{///初始化
    NX = cnt = 0;

memset(Head, -1, sizeof(Head));
    memset(Mx, false, sizeof(Mx));
    memset(My, false, sizeof(My));
}
void AddEdge(int u, int v)
{///添加边
    e[cnt].v = v;
    e[cnt].next = Head[u];
    Head[u] = cnt++;
}
bool BFS()
{///广搜求出层次,并且判断是否有增广路存在
    queue<int> Q;
    depth = oo;

memset(dx, false, sizeof(dx));
    memset(dy, false, sizeof(dy));

for(int i=1; i<=NX; i++)
    {
        if( Mx[i] == false )
        {
            dx[i] = true;
            Q.push(i);
        }
    }

while(Q.size())
    {
        int u = Q.front(); Q.pop();

if( dx[u] > depth )break;///已经发现上层存在增广路

for(int j=Head[u]; j!=-1; j=e[j].next)
        {
            int v = e[j].v;
            if( dy[v] == false )
            {
                dy[v] = dx[u] + 1;

if(My[v] == false)
                    depth = dy[v];
                else
                {
                    dx[ My[v] ] = dy[v] + 1;
                    Q.push( My[v] );
                }
            }
        }
    }

return depth != oo;
}
bool DFS(int i)
{
    for(int j=Head[i]; j!=-1; j=e[j].next)
    {
        int v = e[j].v;

if( used[v] == false && dx[i] == dy[v]-1 )
        {
            used[v] = true;///开始忘记置为true,错了一次
            
            if( My[v] && dy[v] == depth )
                continue;
            if( !My[v] || DFS(My[v]))
            {
                My[v] = i;
                Mx[i] = v;

return true;
            }
        }
    }

return false;
}
int  Karp()
{
    int ans = 0;

while( BFS() == true )
    {
        memset(used, false, sizeof(used));
        for(int i=1; i<=NX; i++)
        {
            if( !Mx[i] && DFS(i) )
                ans++;
        }
    }

///因为点同时在两边,所以会重复一次,结果应该出去 2
    return ans / 2;
}

int main()
{
    int i, j, T, t=1;

scanf("%d", &T);

while(T--)
    {
        scanf("%d", &N);

InIt();

for(i=0; i<N; i++)
        {
            scanf("%s", G[i]);
            for(j=0; j<N; j++)
            {
                if(G[i][j] == ‘#‘)
                {///购置关系图
                    Index[i][j] = ++NX;///给石油编号

if(j != 0 && G[i][j-1] == ‘#‘)
                    {///与左边的可以匹配
                        AddEdge(Index[i][j], Index[i][j-1]);
                        AddEdge(Index[i][j-1], Index[i][j]);
                    }

if(i != 0 && G[i-1][j] == ‘#‘)
                    {///与上面的可以匹配
                        AddEdge(Index[i][j], Index[i-1][j]);
                        AddEdge(Index[i-1][j], Index[i][j]);;
                    }
                }
            }
        }

int ans = Karp();

printf("Case %d: %d\n", t++, ans);
    }

return 0;

}

时间: 2024-08-03 16:58:35

G - Oil Skimming - hdu 4185(二分图匹配)的相关文章

Oil Skimming HDU - 4185(匹配板题)

Oil Skimming Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submission(s): 3426    Accepted Submission(s): 1432 Problem Description Thanks to a certain "green" resources company, there is a new profitabl

HDU 4185 Oil Skimming(离散化 + 二分图匹配)

#include <iostream> #include <cstring> #include <cstdlib> #include <cstdio> #include <vector> #include <queue> #include <stack> #include <cmath> #include <algorithm> #define LL long long #define FOR(i,

hdu 4185 二分图匹配

题意用1*2的木板覆盖矩阵中的‘#’,(木板要覆盖的只能是‘#’),问最多能用几个木板覆盖 将#抽象为二分图的点,一个木板就是一个匹配,注意最后结果要除以2 Sample Input 1 6 ...... .##... .##... ....#. ....## ...... Sample Output Case 1: 3 1 #include<stdio.h> 2 #include<string.h> 3 const int MAXN=1000; 4 char map[610][6

Oil Skimming HDU - 4185

匈牙利 #include <iostream> #include <cstdio> #include <cstring> using namespace std; char mp[660][660]; int g[660][660]; int relate[660][660]; int match[660]; int vis[660]; int k,n; int tmp; int dfs(int u){ int i; for(i = 0; i < tmp; i++

hdu 1281 二分图匹配

题目:在保证尽量多的“车”的前提下,棋盘里有些格子是可以避开的,也就是说,不在这些格子上放车,也可以保证尽量多的“车”被放下.但是某些格子若不放子,就 无法保证放尽量多的“车”,这样的格子被称做重要点.Gardon想让小希算出有多少个这样的重要点,你能解决这个问题么? 二分图匹配居然还能这么用!!!脑洞大开啊!!! 思路:把棋盘的行x看成二分图左边的点,列y看成二分图右边的点,那么就把可以放车的位置看成是一条边,而二分图的最大匹配中x互不相同,y 互不相同,所以每个匹配都是不同行不同列,所以最大

hdu 2063 二分图匹配

题意:一些女的和一些男的有好感,有好感的能一起坐过山车,问最多能组成多少对 hdu 11 页上少有的算法题,二分图匹配问题,匈牙利算法,对于每一个汉子,看和他有好感的妹子有没有配对了,没有配对过就可以成功配对,若已经配对过了,就看那个妹子所配对的汉子能不能再找个没有配对的妹子,如果可以就拆散当前配对重组配对,否则就不能拆:如果找完所有有好感的妹子仍然没能配对成功,那这个汉子就注定孤独一生了``` 1 #include<stdio.h> 2 #include<string.h> 3

过山车 HDU 2063 (二分图匹配裸题)

Problem Description RPG girls今天和大家一起去游乐场玩,终于可以坐上梦寐以求的过山车了.可是,过山车的每一排只有两个座位,而且还有条不成文的规矩,就是每个女生必须找个个男生做partner和她同坐.但是,每个女孩都有各自的想法,举个例子把,Rabbit只愿意和XHD或PQK做partner,Grass只愿意和linle或LL做partner,PrincessSnow愿意和水域浪子或伪酷儿做partner.考虑到经费问题,boss刘决定只让找到partner的人去坐过山

Land of Farms HDU - 5556 二分图匹配

Farmer John and his brothers have found a new land. They are so excited and decide to build new farms on the land. The land is a rectangle and consists of N×MN×Mgrids. A farm consists of one or more connected grids. Two grids are adjacent if they sha

Girls and Boys HDU - 1068 二分图匹配(匈牙利)+最大独立集证明

最大独立集证明参考:https://blog.csdn.net/qq_34564984/article/details/52778763 最大独立集证明: 上图,我们用两个红色的点覆盖了所有边.我们证明的前提条件是已经达到最小覆盖. 即条件1.已经覆盖所有边,条件2.所用的点数最小 首先我们来证明蓝色点组成的是一个独立集:如果有两个蓝色点间有边相连,那么这条 边则没有被覆盖,则与条件1矛盾.因此是独立集. 再来证明这个独立集最大: 如果我们要再增加这个独立集中的点,则需要把某个红点变 成蓝点.而