zoukankan      html  css  js  c++  java
  • Gym 102346A Artwork dfs

    Artwork Gym - 102346A 

    题意:给n*m的地图,入口是(0,0),出口是(n,m),其中有k个监视器,坐标是(xi,yi),监视半径是r,问一个人能不能不被监视到,从起点到终点。

    如果不能走到终点,无非便是监视范围把路全堵死了,所以开始的想法便是计算几何题,看多个圆能不能把横的和竖的都给覆盖了,但很明显行不通,然后想了想觉得这可以转换成一个图的题。

    我们把每个监视器看出点,然后遍历每个监视器跟它相交(相切,包含)的其他点,已经看一下这个圆会涉及到那些边界。然后会把路封死的组合便是,上边界跟左边界,上边界跟下边界,右边界跟下边界,右边界跟左边界。

    (也就是上图中五颜六色的线)

    所以我们就看每个连通块涉及的边界组合中有没有上述边界。

     1 #include<cstdio>
     2 #include<algorithm>
     3 using namespace std;
     4 const int N=1e3+11,M=N*N+11;
     5 struct Side{
     6     int v,ne;
     7 }S[M<<1];
     8 bool vis[N],book[10];
     9 int n,m,k,sn,head[N],xx[N],yy[N],rr[N];
    10 void init(){
    11     sn=0;
    12     for(int i=0;i<k;i++){
    13         head[i]=-1;
    14         vis[i]=false;
    15     }
    16 }
    17 void add(int u,int v){
    18     S[sn].v=v;
    19     S[sn].ne=head[u];
    20     head[u]=sn++;
    21 }
    22 int pf(int x){
    23     return x*x;
    24 }
    25 bool judge(int x,int y){
    26     return pf(xx[x]-xx[y])+pf(yy[x]-yy[y])<=pf(rr[x]+rr[y]);
    27 }
    28 void dfs(int u){
    29     vis[u]=true;
    30     //0 1 2 3分别代表上 右 下 左 边界 
    31     if(xx[u]-rr[u]<=0) book[0]=true;
    32     if(yy[u]+rr[u]>=m) book[1]=true;
    33     if(xx[u]+rr[u]>=n) book[2]=true;
    34     if(yy[u]-rr[u]<=0) book[3]=true;
    35     for(int i=head[u],v;~i;i=S[i].ne){
    36         v=S[i].v;
    37         if(!vis[v]) dfs(v);
    38     }
    39 }
    40 bool solve(){
    41     for(int i=0;i<k;i++){
    42         if(vis[i]) continue;
    43         for(int j=0;j<4;j++) book[j]=false;
    44         dfs(i);
    45         if((book[0]||book[1])&&(book[2]||book[3])) return false;
    46     }
    47     return true;
    48 }
    49 int main(){
    50     while(~scanf("%d%d%d",&n,&m,&k)){
    51         init();
    52         for(int i=0;i<k;i++) scanf("%d%d%d",&xx[i],&yy[i],&rr[i]);
    53         for(int i=0;i<k;i++)
    54             for(int j=i+1;j<k;j++) if(judge(i,j)){
    55                 add(i,j);
    56                 add(j,i);
    57             }
    58         if(solve()) printf("S
    ");
    59         else printf("N
    ");
    60     }
    61     return 0;
    62 }
    小小年纪不学好
  • 相关阅读:
    [Windows] 重新安装/卸载桌面版OneDrive / Reinstall/ Uninstall Desktop Version OneDrive
    [Linux] 关闭防火墙以及开放端口
    [Java] Properties类
    [Linux] 文档编辑搜索
    [Dababase
    etymological
    [JavaScript] 表单验证不通过不提交的JS写法
    Lyrics来源
    [Maven
    [ Servlet / JSP ] J2EE Web Application 中的 JSESSIONID 是什么?
  • 原文地址:https://www.cnblogs.com/LMCC1108/p/11714924.html
Copyright © 2011-2022 走看看