zoukankan      html  css  js  c++  java
  • Codeforces Round #208 E. Dima and Kicks

                                                                           E. Dima and Kicks
                                                                 time limit per test 2 seconds
                                                                 memory limit per test   256 megabytes
    input
    standard input
    output
    standard output

    Dima is a good person. In fact, he's great. But all good things come to an end...

    Seryozha is going to kick Dima just few times.. For this reason he divides the room into unit squares. Now the room is a rectanglen × m consisting of unit squares.

    For the beginning, Seryozha put Dima in a center of some square. Then he started to kick Dima (it is known, that he kicks Dima at least once). Each time when Dima is kicked he flyes up and moves into one of four directions (up, left, right, down). On each move Dima passes k(k > 1) unit of the length in the corresponding direction. Seryozha is really kind, so he kicks Dima in such way that Dima never meets the walls (in other words, Dima never leave the room's space). Seryozha is also dynamic character so Dima never flies above the same segment, connecting a pair of adjacent squares, twice.

    Seryozha kicks Dima for a long time, but Dima is not vindictive — Dima writes. Dima marked all squares in which he was staying or above which he was flying. Thanks to kicks, Dima does not remember thek value, so he asks you to find all possible values which matches to the Dima's records.

    Input

    The first line contains n andm (1 ≤ n, m ≤ 103) — size of the room.

    Next n lines goes, each containsm numbers aij — Dima's notes:aij = 1, if Dima was staying in the square(i, j) or was flying above it. Otherwiseaij = 0.

    At least one aij equals1.

    Output

    In a single line in accending order print all k (k > 1), which matches the Dima's notes. If there are no suchk and Dima invented this story with kicks, print -1.


    看了别人的代码,思路至今仍未捋清。

    在2和Min(n,m)之间枚举k值,遍历矩阵,多条件检测k值。(以后再详解。。。)

    #include <stdio.h>
    #include <stdlib.h>
    #include <algorithm>
    #include <string.h>
    #include <iostream>
    #include <map>
    #include <vector>
    #include <queue>
    #include <set>
    #include <string>
    #include <math.h>
    #define N 1010
    using namespace std;
    int n,m,tot;
    int a[N][N];
    int s0[N][N],s1[N][N],d[N][N];
    int readInt(){        //输入挂
      char c;
      int n=0;
      c=getchar();
      while(c<'0'||c>'9') c=getchar();
      while(c>='0'&&c<='9'){
        n=n*10+c-'0';
        c=getchar();                      
      }    
      return n;
    }
    int fa[N*N];
    int sx,sy;   //默认初始化为0
    int ok(int x,int y){
      return x>=1&&x<=n&&y>=1&&y<=m ;    
    }
    int find(int i)      //并查集
    {
      if(fa[i]==i)  return i;
      else  return find(fa[i]);   
    }
    int can(int k){
        int i,j,x,y,mx,my,tmp;
        int dd=0,fmx;
        fmx=(sx-1)/k,mx=(n-sx)/k,my=(m-sy)/k,dd=0;   //mx,my记录距离下墙、右墙的kick数
        for(i=-fmx;i<=mx;i++)
          for(j=0;j<=my;j++){
             d[i+fmx][j]=0;
             int num=(i+fmx)*(my+1)+j;
             fa[num]=num;
          }
        for(i=-fmx;i<=mx;i++){
          for(j=0;j<=my;j++){
            x=sx+i*k;     //所能到达的位置
            y=sy+j*k;                   
            if(!a[x][y])  continue;
            dd++;
            if(ok(x+k,y)&&a[x+k][y]){   //向下走
              tmp=s0[x+k-1][y]-s0[x][y];
              if(tmp==k-1){
                d[i+fmx][j]++;
                d[i+1+fmx][j]++;
                dd+=k-1;
                fa[find((i+fmx)*(my+1)+j)]=fa[find((i+1+fmx)*(my+1)+j)];             
              }    
              else if(tmp>0)  return 0;                   
            }
            if(ok(x,y+k)&&a[x][y+k]){  //向右走
              tmp=s1[x][y+k-1]-s1[x][y];
              if(tmp==k-1){
                d[i+fmx][j]++;
                d[i+fmx][j+1]++;
                dd+=k-1;
                fa[find((i+fmx)*(my+1)+j)]=fa[find((i+fmx)*(my+1)+j+1)];             
              }    
              else if(tmp>0)  return 0;                          
            }        
          }                      
        }
        if(dd!=tot)  return 0;        //所经过的单元总数
        int cnt=0,thef=find((0+fmx)*(my+1)+0);
        for(i=-fmx;i<=mx;i++)
          for(j=0;j<=my;j++){
            cnt+=d[i+fmx][j]&1;                   
          }
        for(i=-fmx;i<=mx;i++)
          for(j=0;j<=my;j++){
            x=sx+i*k;
            y=sy+j*k;
            if(!a[x][y])  continue;
            if(find((i+fmx)*(my+1)+j)!=thef)   return 0;
          }
        return cnt<=2;
    }
    int main()
    {
      int i,j,k;
      while(scanf("%d %d",&n,&m)!=EOF){
        tot=0;
        for(i=1;i<=n;i++)
          for(j=1;j<=m;j++){  
            //scanf("%d",&a[i][j]);
            a[i][j]=readInt();
            tot+=a[i][j];         //通过的总单元数            
          }
        if((n<3&&m<3)|| tot==1){
           printf("-1 ");             
        }
        else{
         for(i=1;i<=n;i++)
           for(j=1;j<=m;j++){  
            s0[i][j]=s0[i-1][j]+a[i][j];
            s1[i][j]=s1[i][j-1]+a[i][j];            
           }
         sx=sy=0;
         for(j=1;j<=m;j++){      //纵向搜索第一个不为0的单元位置
           for(i=1;i<=n;i++){
            if(a[i][j]){
              sx=i;sy=j; break;             
            }                  
           }                  
           if(sx) break;
         }
         int flag=1;
         for(k=2;k<=min(n,m);k++){
               //printf("can(%d)=%d ",k,can(k));
               if(can(k)){
                 if(flag)  flag=0;
                 else   printf(" ");
                 printf("%d",k);         
               }                                                            
         }
         if(flag)  printf("-1 ");
         else printf(" ");
        }         
      }
      return 0;
    }


  • 相关阅读:
    POJ2888 Magic Bracelet [矩阵快速幂+Burnside+欧拉函数]
    数列的 GCD [计数问题]
    com组件的注册
    WCF 传输和接受大数据
    数据库中已存在名为 'View_Business' 的对象。
    windows 两个用户,默认其中一个用户登录
    用C#读取,写入ini文件
    小心得,关于串口
    未能加载文件或程序集"Microsoft.Web.Infrastructure, Version=1.0.0.0, Culture=neutral, PublicKeyToken=31bf3856ad
    无法激活服务,因为它不支持 ASP.NET 兼容性
  • 原文地址:https://www.cnblogs.com/james1207/p/3395247.html
Copyright © 2011-2022 走看看