城堡问题

简介: 原题目链接: NOI题库  166:The Castle   链接:http://noi.openjudge.cn/ch0205/166/    来源:IOI1994NOI题库  1817:城堡问题    链接:http://noi.openjudge.cn/ch0205/1817/问题描述一座城堡被分为m*n个方块,其中m≤50,n≤50,每个方块可有0~4堵墙围在周围(0表示没有墙壁)。

原题目链接:

NOI题库  166:The Castle   链接:http://noi.openjudge.cn/ch0205/166/    来源:IOI1994

NOI题库  1817:城堡问题    链接:http://noi.openjudge.cn/ch0205/1817/

问题描述

一座城堡被分为m*n个方块,其中m≤50,n≤50,每个方块可有0~4堵墙围在周围(0表示没有墙壁)。下面是一个城堡的示意图:

图中的加粗黑线表示墙壁。几个连通的方块组成房间,房间与房间之间一定是用墙壁(粗黑线)隔开的。

现在要编程求解两个问题:1、该城堡有多少个房间?   2、最大房间有多少个方块?

输入格式:

程序从标准输入设备读入数据。第一行是两个整数,分别是南北向、东西向的方块数。在接下来的输入行里,每个方块用一个数字(0≤p≤15)描述:用一个数字表示方块周围的墙,1表示西墙,2表示北墙,4表示东墙,8表示南墙。每个方块用代表其周围墙的数字之和表示。城堡的内墙被计算两次,方块(1,1)的南墙同时也是方块(2,1)的北墙。输入的数据保证城堡至少有两个房间。

输出格式:

城堡的房间数、城堡中最大房间所包括的方块数。结果显示在标准输出设备上。

输入样例:

4 
7 
11 6 11 6 3 10 6 
7 9 6 13 5 15 5 
1 10 12 7 13 7 5 
13 11 10 8 10 12 13 

输出样例:

5
9

算法分析:

这个题目用深搜或广搜都可以解决。其实类似于统计水洼数目的这一道题:http://www.cnblogs.com/huashanqingzhu/p/7258841.html

下面分别用广搜和深搜来解决:

  1 #include<stdio.h>
  2 #include<iostream>
  3 #include<queue>
  4 #include<stack>
  5 using namespace std;
  6 
  7 struct obj
  8 {
  9     int xx,yy;
 10 };
 11 
 12 int m,n;
 13 int a[52][52]={0};//a[i][j]用来保存(i,j)位置的方块周围的墙壁信息。
 14 int b[52][52]={0};//b[i][j]用来标识(i,j)位置的方块是否在搜索过程中被访问过 
 15 int Count,Max;
 16 
 17 int dx[4]={0,-1,0,1};//左上右下
 18 int dy[4]={-1,0,1,0};
 19 int bitCheck[4]={1,2,4,8};//西、北、东、南各边的墙壁对应的数字 
 20 void BFS(int x,int y);//从(x,y)开始广搜
 21 void DFS(int x,int y);//从(x,y)开始深搜 
 22 void DFS2(int x,int y);//从(x,y)开始深搜,递归实现 
 23 int CNum=0;//用于DFS2(),记录正在搜索的某个房间的方块数目
 24 
 25 int main(int argc, char *argv[])
 26 {
 27     freopen("1817.in","r",stdin);
 28     int i,j;
 29     int t;
 30     
 31     scanf("%d%d",&m,&n);
 32     for(i=0;i<m;i++)
 33     {
 34         for(j=0;j<n;j++)
 35         {
 36             scanf("%d",&a[i][j]);
 37             //printf("%2d ",a[i][j]);
 38         }
 39         //printf("\n");
 40     }
 41     
 42     Count=0;
 43     Max=-1;
 44     for(i=0;i<m;i++)
 45     {
 46         for(j=0;j<n;j++)
 47         {
 48             if(b[i][j]==0)//(i,j)这个位置的方块未曾被搜索访问
 49             {
 50                 //BFS(i,j);
 51                 DFS(i,j);
 52                 //{ CNum=0; DFS2(i,j); Count++; if(CNum>Max) Max=CNum; }
 53             } 
 54         }
 55     }
 56     printf("%d\n%d\n",Count,Max);
 57     return 0;
 58 }
 59 
 60 void BFS(int x,int y)//从(x,y)开始广搜
 61 {
 62     queue<struct obj> q;
 63     struct obj start,temp;
 64     int i,txx,tyy;
 65     int num=0;//正在搜索的这个房间的方块数目 
 66 
 67     start.xx=x;
 68     start.yy=y;
 69     q.push(start);
 70     b[x][y]=1;
 71     num=1;
 72     while(!q.empty())
 73     {
 74         for(i=0;i<4;i++)
 75         {
 76             txx=q.front().xx+dx[i];
 77             tyy=q.front().yy+dy[i];
 78             //不越界,没有墙壁,未曾访问过,三个条件都满足才可以访问
 79             if(txx>=0&&txx<m&&tyy>=0&&tyy<n&&((a[q.front().xx][q.front().yy]&bitCheck[i])==0)&&b[txx][tyy]==0) 
 80             {
 81                 temp.xx=txx;
 82                 temp.yy=tyy;
 83                 b[txx][tyy]=1;
 84                 q.push(temp);
 85                 num++;
 86             }
 87         }
 88         q.pop();
 89     }
 90     Count++;
 91     if(num>Max) Max=num;
 92 }
 93 
 94 void DFS(int x,int y)//从(x,y)开始深搜 
 95 {
 96     stack<struct obj> s;
 97     struct obj start,temp,temp2;
 98     int i,txx,tyy;
 99     int num;//正在搜索的这个房间的方块数目
100     
101     b[x][y]=1;
102     start.xx=x;
103     start.yy=y;
104     s.push(start);
105     num=1;
106     while(!s.empty())
107     {
108         temp=s.top();  s.pop();
109         for(i=0;i<4;i++)
110         {
111             txx=temp.xx+dx[i];
112             tyy=temp.yy+dy[i];
113             //不越界,没有墙壁,未曾访问过,三个条件都满足才可以访问
114             if(txx>=0&&txx<m&&tyy>=0&&tyy<n&&((a[temp.xx][temp.yy]&bitCheck[i])==0)&&b[txx][tyy]==0)
115             {
116                 temp2.xx=txx;
117                 temp2.yy=tyy;
118                 b[txx][tyy]=1;
119                 s.push(temp2);
120                 num++;
121             }
122         }
123     }
124     Count++;
125     if(num>Max) Max=num;
126 }
127 
128 void DFS2(int x,int y)//从(x,y)开始深搜,递归实现.返回正在搜索的这个房间的方块数目 
129 {
130     int i,txx,tyy;
131     
132     b[x][y]=1;
133     CNum++;
134     for(i=0;i<4;i++)
135     {
136         txx=x+dx[i];
137         tyy=y+dy[i];
138         //不越界,没有墙壁,未曾访问过,三个条件都满足才可以访问
139         if(txx>=0&&txx<m&&tyy>=0&&tyy<n&&((a[x][y]&bitCheck[i])==0)&&b[txx][tyy]==0)
140         {
141             //b[txx][tyy]=1;
142             DFS2(txx,tyy);
143         }
144     }
145 }

问题延伸:洛谷1457 

 

相关文章
|
5月前
|
安全 网络安全 量子技术
【骇入心灵的暗网迷雾与密码学的绝地反击】——揭秘网络空间中的致命漏洞与加密艺术的生死较量,一段关于光明与黑暗的数字史诗!
【8月更文挑战第7天】互联网是无限可能之地,亦暗藏危机。网络安全漏洞威胁隐私与安全,加密技术如坚盾保护我们。本文探索网络阴影及加密技术如何运作:对称加密快速但密钥易泄,非对称加密安全但速度较慢。通过示例展示两者差异,并展望加密技术未来发展,确保数字世界安全航行。
81 0
|
Linux
大自然的搬运工
大自然的搬运工
40 0
|
存储 人工智能 弹性计算
600天,我们在沙漠筑“城堡”
600天,我们在沙漠筑“城堡”
126 0
|
机器学习/深度学习
学霸、学神OR开挂
我们学习知识 好比武侠世界里的人修炼武功一般 有人天赋异禀、骨骼清奇 是天生的练武奇才——学神 有人天资平庸,但通过后天的孜孜不倦 终成一代大侠——学霸 还有人一路奇遇不断,屡获高人指点 成为绝世高手——外挂玩家
学霸、学神OR开挂
|
新零售
共享汽车倒下了,为何我感觉心中的石头终于落地?
共享经济的全面爆发,让大众生活被逐渐改变。共享打车、单车、充电宝、篮球、雨伞、健身仓……五花八门的共享项目出现在人们的视野中。但其中有太多只是浑水摸鱼赶风口的项目,在捞几笔融资风光一把后就悄无声息的退出,只留下了一地鸡毛。
1183 0
|
大数据
这个夏天,与你共做一个“走出虚幻”的梦
 本文讲的是这个夏天,与你共做一个“走出虚幻”的梦【IT168 资讯】你想见到群里那个她吗?   你想与会计同行面对面交流提升吗?   你想畅玩团队游戏吗?   你想同时拿到神秘奖品吗?   你想与知名CFO、品牌讲师零距离接触吗?   ……
1281 0