No 4.5 宝岛探险

简单介绍:在一个二维格子上面的一点,其相邻的格子:0表示海洋,1~9表示陆地,求一个目标格子所在的岛面积。

注意:与目标点上下左右相链接的陆地视为同一岛屿,不纠结其他四个角!

一、广度优先 VS 深度优先

/* No.1 广度优先搜索
#include <stdio.h>
struct node{  //广度优先,一定有一个队列用来存储父子节点
  int x;
  int y;
};
int book[50][50]={0};

int main(){
  struct node que[100];
  int head=1,tail=1;

  int m[50][50];
  int i,j,k,sum=0;
  int n,m,p,q,tx,ty;

  int next[4][2]={
    {0,1},{1,0},{0,-1},{-1,0} //{1,1},{1,-1},{-1,-1},{-1,1}
  };

  scanf("%d %d %d %d",&n,&m,&p,&q);  //犯了个错,scanf里面加了",",输入的时候没加!
  for(i=1;i<=n;i++)
    for(j=1;j<=m;j++)
      scanf("%d",&m[i][j]);      //与 char m[i][j]区别

  if(m[p][q]!=0)
  {
    que[tail].x=p;
    que[tail].y=q;
    sum++;
    book[p][q]=1;
    tail++;
  }

  while(head<tail){
    for(k=0;k<3;k++){
      tx=que[head].x+next[k][0];
      ty=que[head].y+next[k][1];
      if(tx<1||tx>10||ty<1||ty>10)
        continue;
      if(m[tx][ty]!=0 && book[tx][ty]==0){
        sum++;
        book[tx][ty]=1;
        que[tail].x=tx;
        que[tail].y=ty;
        tail++;
      }
    }
    head++;  //一定要记住,父节点的出栈
  }
  printf("%d",sum);
  getchar();getchar();
  return 0;
}
*/

/* No.4 深度优先搜索
#include <stdio.h>
int book[50][50];  //边界大点,否则结果可能不准确
int n,m,p,q,sum;
int map[50][50];
void dfs(int x,int y){
  int k,tx,ty;
  int next[4][2]={
    {0,1},{1,0},{0,-1},{-1,0} //{1,1},{1,-1},{-1,-1},{-1,1}
  };
  for(k=0;k<=3;k++){
    tx=x+next[k][0];
    ty=y+next[k][1];
    if(tx<0 ||tx>n || ty<0 ||ty>m)
      continue;
    if(map[tx][ty]!=0 && book[tx][ty]==0){
      sum++;
      book[tx][ty]=1;
      dfs(tx,ty);
    }
  }
  return ;  //千万不要忘了这一句,返回上一层dfs
}

int main(){
  int i,j;
  scanf("%d %d %d %d",&n,&m,&p,&q);
  for(i=1;i<=n;i++)
    for(j=1;j<=m;j++)
      scanf("%d",&map[i][j]);
  if(map[p,q]!=0){
    sum++;
    book[p][q]=1;
    dfs(p,q);
  }
  printf("%d",sum);
  getchar();getchar();
  return 0;
}
*/

 

二、更进一步:要求将上述岛标记出来

int book[50][50];
int n,m,p,q;
int map[50][50];
void dfs(int x,int y,int color){
  int k,tx,ty;
  int next[4][2]={
    {0,1},{1,0},{0,-1},{-1,0} //{1,1},{1,-1},{-1,-1},{-1,1}
  };
  map[x][y]=color;  //添加一个颜色标记就好了
  for(k=0;k<=3;k++){
    tx=x+next[k][0];
    ty=y+next[k][1];
    if(tx<0 ||tx>n || ty<0 ||ty>m)
      continue;
    if(map[tx][ty]!=0 && book[tx][ty]==0){
      book[tx][ty]=1;
      dfs(tx,ty,color);
    }
  }
  return ;
}

int main(){
  int i,j;
  scanf("%d %d %d %d",&n,&m,&p,&q);
  for(i=1;i<=n;i++)
    for(j=1;j<=m;j++)
      scanf("%d",&map[i][j]);
  if(map[p,q]!=0){
    book[p][q]=1;
    dfs(p,q,-1);
  }

  for(i=1;i<=n;i++)  //下面的两个 {} 不能省略
  {  

    for(j=1;j<=m;j++)
      {printf("%3d",map[i][j]);}
    printf("\n");
  }
  getchar();getchar();
  return 0;
}

 

三、再进一步:把图中独立的小岛,分别标记出来

#include <stdio.h>
int book[50][50]={0};
int n,m,p,q;
int map[50][50];
void dfs(int x,int y,int color){
  int k,tx,ty;
  int next[4][2]={
    {0,1},{1,0},{0,-1},{-1,0} //{1,1},{1,-1},{-1,-1},{-1,1}
  };

  map[x][y]=color;
  for(k=0;k<=3;k++){
    tx=x+next[k][0];
    ty=y+next[k][1];
    if(tx<0 ||tx>n || ty<0 ||ty>m)
      continue;
    if(map[tx][ty]!=0 && book[tx][ty]==0){
      book[tx][ty]=1;
      dfs(tx,ty,color);
    }
  }
  return ;
}

int main(){
  int i,j,color=0;
  scanf("%d %d",&n,&m);
  for(i=1;i<=n;i++)
    for(j=1;j<=m;j++)
      scanf("%d",&map[i][j]);

  for(i=1;i<=n;i++)
    for(j=1;j<=m;j++)
    {
      if(map[i][j]>0)  // 这里居然不能继续添加 && book[i][j]=0 条件
      {
        color--;
        book[i][j]=1;
        dfs(i,j,color);
      }
    }

  for(i=1;i<=n;i++)
  {

     for(j=1;j<=m;j++)
      {printf("%3d",map[i][j]);}
    printf("\n");
  }
  printf("共有%d个小岛",-color);
  getchar();getchar();
  return 0;
}

上一篇:(2) 关于USB3.0


下一篇:解救小哈