【最小路径覆盖】BZOJ2150-部落战争

【题目大意】

给出一张图,'*'表示不能走的障碍。已知每只军队可以按照r*c的方向行军,且军队与军队之间路径不能交叉。问占据全部'.'最少要多少支军队?

【思路】

首先注意题意中有说“军队只能往下走”,弄清楚方向。

从某点往它能走的四个点走一趟,连边。最小路径覆盖=总数-二分图最大匹配。

哦耶!老了,连匈牙利的板子都敲错orzzzzzz

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int MAXN=55;
 4 int m,n,r,c,maps[MAXN][MAXN];
 5 int lk[MAXN*MAXN],vis[MAXN*MAXN];
 6 vector<int> E[MAXN*MAXN];
 7 
 8 int check(int x,int y)
 9 {
10     if (x>=1 && x<=m && y>=1 && y<=n && maps[x][y]) return 1;else return 0;
11 }
12 
13 int id(int x,int y){return ((x-1)*n+y);}
14 
15 void addedge(int u,int v)
16 {
17     E[u].push_back(v);
18 }
19 
20 int find(int u)
21 {
22     for (int i=E[u].size()-1;i>=0;i--)
23     {
24         int v=E[u][i];
25         if (!vis[v])
26         {
27             vis[v]=1;
28             if (!lk[v]|| find(lk[v]))//呜哇这里写成了find(v)半天没有发现,果然是老阿姨了啊 
29             {
30                 lk[v]=u;
31                 return 1;
32             }
33         }
34     }
35     return 0;
36 }
37 
38 void init()
39 {
40     scanf("%d%d%d%d",&m,&n,&r,&c);
41     for (int i=1;i<=m;i++)
42     {
43         char str[MAXN];
44         scanf("%s",str);
45         for (int j=0;j<n;j++)
46             if (str[j]=='.') maps[i][j+1]=1;else maps[i][j+1]=0;
47     }
48     for (int i=1;i<=m;i++)
49         for (int j=1;j<=n;j++)
50             if (maps[i][j])
51             {
52                 if (check(i+r,j+c)) addedge(id(i,j),id(i+r,j+c));
53                 if (check(i+r,j-c)) addedge(id(i,j),id(i+r,j-c));
54                 if (check(i+c,j+r))    addedge(id(i,j),id(i+c,j+r));
55                 if (check(i+c,j-r)) addedge(id(i,j),id(i+c,j-r));
56             }
57 }
58 
59 void solve()
60 {
61     memset(lk,0,sizeof(lk));
62     int sum=0,ans=0;
63     for (int i=1;i<=m;i++)
64         for (int j=1;j<=n;j++)
65         {
66             if (maps[i][j]==1)
67             {
68                 int x=id(i,j);
69                 memset(vis,0,sizeof(vis));
70                 sum++;
71                 if (find(x)) ans++;
72             }
73         }
74     printf("%d",sum-ans);
75 }
76 
77 int main()
78 {
79     init();
80     solve();
81 }

 

posted @ 2016-10-02 22:31  iiyiyi  阅读(215)  评论(0编辑  收藏  举报