zoukankan      html  css  js  c++  java
  • 【最大独立集】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 }
  • 相关阅读:
    Redis为什么使用单进程单线程方式也这么快
    数据库表设计五大范式所解决的问题
    数据库范式那些事
    线上Java程序导致服务器CPU占用率过高的问题排除过程
    对分布式事务及两阶段提交、三阶段提交的理解
    About the diffrence of wait timed_wait and block in java
    Linux io Model
    阻塞与非阻塞,同步与异步
    深入理解JAVA I/O系列六:Linux中的IO模型(转载的文章非常值得学习)
    真正的Maven经常使用命令
  • 原文地址:https://www.cnblogs.com/iiyiyi/p/5928351.html
Copyright © 2011-2022 走看看