【最大独立集】BZOJ3175- [Tjoi2013]攻击装置

【题目大意】

给定一个01矩阵,其中你可以在0的位置放置攻击装置。每一个攻击装置(x,y)都可以按照“日”字攻击其周围的 8个位置(x-1,y-2),(x-2,y-1),(x+1,y-2),(x+2,y-1),(x-1,y+2),(x-2,y+1), (x+1,y+2),(x+2,y+1)。
求在装置互不攻击的情况下,最多可以放置多少个装置。

【思路】

最大独立集=总点数-最大二分图匹配

我们都知道最大独立集一般用在二分图里…所以把每个点看作两个点,一个是从它出发,一个是抵达它。实际操作的时候因为会连正反边,不需要进行拆点。最大独立集=总点数-最大二分图匹配/2。

 1 #include<bits/stdc++.h>
 2 using namespace std;
 3 const int MAXN=205;
 4 int n,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<=n && 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]))
29             {
30                 lk[v]=u;
31                 return 1;
32             }
33         }
34     }
35     return 0;
36 }
37
38 void init()
39 {
40     scanf("%d",&n);
41     for (int i=1;i<=n;i++)
42     {
43         char str[MAXN];
44         scanf("%s",str);
45         for (int j=0;j<n;j++)
46             if (str[j]==‘0‘) maps[i][j+1]=1;else maps[i][j+1]=0;
47     }
48     for (int i=1;i<=n;i++)
49         for (int j=1;j<=n;j++)
50             if (maps[i][j]==1)
51             {
52                 if (check(i-1,j-2)) addedge(id(i,j),id(i-1,j-2));
53                 if (check(i-2,j-1)) addedge(id(i,j),id(i-2,j-1));
54                 if (check(i+1,j-2)) addedge(id(i,j),id(i+1,j-2));
55                 if (check(i+2,j-1)) addedge(id(i,j),id(i+2,j-1));
56                 if (check(i-1,j+2)) addedge(id(i,j),id(i-1,j+2));
57                 if (check(i-2,j+1)) addedge(id(i,j),id(i-2,j+1));
58                 if (check(i+1,j+2)) addedge(id(i,j),id(i+1,j+2));
59                 if (check(i+2,j+1)) addedge(id(i,j),id(i+2,j+1));
60             }
61 }
62
63 void solve()
64 {
65     memset(lk,0,sizeof(lk));
66     int sum=0,ans=0;
67     for (int i=1;i<=n;i++)
68         for (int j=1;j<=n;j++)
69         {
70             if (maps[i][j]==1)
71             {
72                 int x=id(i,j);
73                 memset(vis,0,sizeof(vis));
74                 sum++;
75                 if (find(x)) ans++;
76             }
77         }
78     printf("%d",sum-ans/2);//不要忘记了除以2
79 }
80
81 int main()
82 {
83     init();
84     solve();
85 }
时间: 2024-08-07 04:33:21

【最大独立集】BZOJ3175- [Tjoi2013]攻击装置的相关文章

bzoj4808: 马 &amp; bzoj3175: [Tjoi2013]攻击装置 (黑白染色+最小割)

bzoj4808: 马 & bzoj3175: [Tjoi2013]攻击装置 题目:传送门 简要题意: 和n皇后问题差不多,但是这里是每个棋子走日子,而且有些格子不能放棋子.求最多能放多少个棋子. 题解: 双倍经验好评 之前看过机房神犇做...有点印象是最小割. 但是直接割的话不会...要应用到黑白染色: 最开始我想递归染色,也就是取一个开始染,然后递归下去... 波老师说会错ORZ...因为感觉递归的情况不好掌握,有可能会重复染色... 结果有一个肥肠巧妙的方法...直接相邻的染为不同颜色 之

BZOJ3175 [Tjoi2013]攻击装置

Description 给 定一个01矩阵,其中你可以在0的位置放置攻击装置.每一个攻击装置(x,y)都可以按照“日”字攻击其周围的 8个位置(x-1,y-2),(x-2,y-1),(x+1,y-2),(x+2,y-1),(x-1,y+2),(x-2,y+1), (x+1,y+2),(x+2,y+1) 求在装置互不攻击的情况下,最多可以放置多少个装置. Input 第一行一个整数N,表示矩阵大小为N*N.接下来N行每一行一个长度N的01串,表示矩阵. Output 一个整数,表示在装置互不攻击的

BZOJ4808: 马&amp;BZOJ3175: [Tjoi2013]攻击装置

BZOJ3175 Description 给定一个01矩阵,其中你可以在0的位置放置攻击装置.每一个攻击装置(x,y)都可以按照“日”字攻击其周围的 8个位置(x-1,y-2),(x-2,y-1),(x+1,y-2),(x+2,y-1),(x-1,y+2),(x-2,y+1), (x+1,y+2),(x+2,y+1)求在装置互不攻击的情况下,最多可以放置多少个装置. Input 第一行一个整数N,表示矩阵大小为N*N.接下来N行每一行一个长度N的01串,表示矩阵. Output 一个整数,表示在

bzoj3175: [Tjoi2013]攻击装置&amp;&amp;4808: 马

终于知道为啥网络流这么受欢迎了. 其实就是构个图模板一下的事儿,比较好打是吧. 然后这题网络流黑白染色(其实感觉上匈牙利更加直接好想啊,但是实际上黑白染色给人感觉就是二分图) st连白而ed连黑,流量为1 不能同时出现的就建无限流量的边 然后sum-最小割 #include<cstdio> #include<iostream> #include<cstring> #include<cstdlib> #include<algorithm> #inc

[TJOI2013] 攻击装置

3175: [Tjoi2013]攻击装置 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1326  Solved: 636[Submit][Status][Discuss] Description 给定一个01矩阵,其中你可以在0的位置放置攻击装置.每一个攻击装置(x,y)都可以按照"日"字攻击其周围的 8个位置(x-1,y-2),(x-2,y-1),(x+1,y-2),(x+2,y-1),(x-1,y+2),(x-2,y+1), (x

BZOJ 3175: [Tjoi2013]攻击装置( 匈牙利 )

黑白染成二分图, 然后不能同时选的就连边, 最大匹配数为m, t为不能放的数目, 则题目所求最大点独立集为 n*n-m-t ------------------------------------------------------------------------- #include<cstdio> #include<cstring> #include<algorithm> using namespace std; const int maxn = 209; con

3175. [TJOI2013]攻击装置【二分图最大独立集】

Description 给定一个01矩阵,其中你可以在0的位置放置攻击装置.每一个攻击装置(x,y)都可以按照“日”字攻击其周围的 8个位置(x-1,y-2),(x-2,y-1),(x+1,y-2),(x+2,y-1),(x-1,y+2),(x-2,y+1), (x+1,y+2),(x+2,y+1) 求在装置互不攻击的情况下,最多可以放置多少个装置. Input 第一行一个整数N,表示矩阵大小为N*N.接下来N行每一行一个长度N的01串,表示矩阵. Output 一个整数,表示在装置互不攻击的情

BZOJ 3175 Tjoi2013 攻击装置 二分图最大匹配

题目大意:给定一个n*n的网格图,要在0的位置上放置一些攻击装置,其中一个攻击装置的攻击范围是周围8个"日"字形区域,要求不能互相攻击,求最多放置多少个攻击装置 每两个能互相攻击且能放置的点连一条双向边,然后跑二分图最大点独立集即可 4W个点n^2居然没TLE 是数据太弱还是匈牙利算法太强了? #include<cstdio> #include<cstring> #include<iostream> #include<algorithm>

【bzoj3175】[Tjoi2013]攻击装置

每两个能互相攻击且能放置的点连一条双向边,然后跑二分图最大点独立集即可 #include<algorithm> #include<iostream> #include<cstdlib> #include<cstring> #include<cstdio> #include<cmath> using namespace std; typedef long long LL; #define M 80010 #define N 210 st