【题解】Reach-Top OJ [1115] 勘探油田

题目描述

某石油勘探公司正在按计划勘探地下油田资源,工作在一片长方形的地域中。他们首先将该地域划分为许多小正方形区域,然后使用探测设备分别探测每一块小正方形区域内是否有油。若在一块小正方形区域中探测到有油,则标记为’’,否则标记为’’。如果两个相邻区域都为,那么它们同属于一个石油带,一个石油带可能包含很多小正方形区域,而你的任务是要确定在一片长方形地域中有多少个石油带。 所谓相邻,是指两个小正方形区域上下、左右、左上右下或左下右上同为’’。

输入

输入数据将包含一些长方形地域数据,每个地域数据的第一行有两个正整数,表示该地域由个小正方形所组成,如果,表示所有输入到此结束;否则,后面)行数据,每行有)个字符,每个字符为’’或’’,表示有油或无油。每个长方形地域中,’’的个数不会超过

输出

每个长方形地域,输出油带的个数,每个油带值占独立的一行。油带值不会超过100。

样例输入

1 1
*
3 5
* @ * @ *
* * @ * *
* @ * @ *
1 8
@ @ * * * * @ *
5 5
* * * * @
* @ * * @
* @ * * @
@ @ @ * @
@ @ * * @
0 0

样例输出

0
1
2
2

思路&解答

经典的连通块题目,使用DFS搜索。
题意让我们求油田的块数,也就是求联通块数。
首先先要清楚搜索方向,我们将当前格定义为,则以此格可推出其他八格分别为:


如下图:
image
故我们的方向函数gox[],goy[]为:

const int gox[] = { 1,-1,0,0,1,-1,1,-1 };
const int goy[] = { 0,0,1,-1,1,-1,-1,1 };

扫描全局见到''时调用DFS(int x,int y)函数,并且计数器sum自增
我们把每次走到的一格都标记为'',把油田覆盖为平地,以免重复计数。
再次覆盖一整块油田,如此反复直到所有的地均为''时结束。
完整的DFS(int x,int y)函数为:

void DFS(int x, int y)
{
    mp[x][y] = '*'; //mp[][]是存放地图的二维数组哦~~
    for (int i = 0; i < 8; i++)
    {
        int next_x = x + gox[i];
        int next_y = y + goy[i];
        if (next_x > 0 && next_y > 0 && next_x <= m && next_y <= n && mp[next_x][next_y] == '@')
            DFS(next_x, next_y); //下一步扫描
    }
}

最后,输出计数器的sum即可。

代码

#include <bits/stdc++.h>
using namespace std;
const int maxn = 105;
const int gox[] = { 1,-1,0,0,1,-1,1,-1 };
const int goy[] = { 0,0,1,-1,1,-1,-1,1 };
char mp[maxn][maxn];
int m, n;
void DFS(int x, int y)
{
    mp[x][y] = '*';
    for (int i = 0; i < 8; i++)
    {
        int next_x = x + gox[i];
        int next_y = y + goy[i];
        if (next_x > 0 && next_y > 0 && next_x <= m && next_y <= n && mp[next_x][next_y] == '@')
            DFS(next_x, next_y);
    }
}
int main()
{
    while (cin >> m >> n && m)
    {
        int sum = 0;
        for (int i = 1; i <= m; i++)
            for (int j = 1; j <= n; j++)
                cin >> mp[i][j];
        for (int i = 1; i <= m; i++)
            for (int j = 1; j <= n; j++)
                if (mp[i][j] == '@')
                {
                    DFS(i, j);
                    sum++;
                }
        cout << sum << endl;
    }
    return 0;
}
全部评论

相关推荐

learYuan:🐕看了都摇头
点赞 评论 收藏
分享
点赞 评论 收藏
分享
从输入URL到页面加载发生了什么:总体来说分为以下几个过程:&nbsp;1.DNS解析&nbsp;2.TCP连接&nbsp;3.发送HTTP请求&nbsp;4.服务器处理请求并返回HTTP报文&nbsp;5.浏览器解析渲染页面&nbsp;6.连接结束。简述了一下各个过程的输入输出作用:以下是对从输入&nbsp;URL&nbsp;到页面加载各过程的输入、输出或作用的一句话描述:DNS&nbsp;解析:&nbsp;输入:用户在浏览器地址栏输入的域名(如&nbsp;www.example.com)。输出:对应的&nbsp;IP&nbsp;地址(如&nbsp;192.168.1.1)。作用:将易于记忆的域名转换为计算机能够识别和用于网络通信的&nbsp;IP&nbsp;地址,以便浏览器与目标服务器建立连接。TCP&nbsp;连接:&nbsp;输入:浏览器获得的服务器...
明天不下雨了:参考一下我的说法: 关键要讲出输入网址后涉及的每一个网络协议的工作原理和作用: 涉及到的网络协议: HTTP/HTTPS协议->DNS协议->TCP协议->IP协议->ARP协议 面试参考回答: 第一次访问(本地没有缓存时): 一般我们在浏览器地址栏输入的是一个域名。 浏览器会先解析 URL、解析出域名、资源路径、端口等信息、然后构造 HTTP 请求报文。浏览器新开一个网络线程发起HTTP请求(应用层) 接着进行域名解析、将域名解析为 IP 地址 浏览器会先检查本地缓存(包括浏览器 DNS 缓存、操作系统缓存等)是否已解析过该域名 如果没有、则向本地 DNS 服务器请求解析; 本地服务器查不到会向更上层的 DNS 服务器(根域名服务器->顶级域名服务器->权威域名服务器询问)递归查询 最终返回该域名对应的 IP 地址。(应用层DNS协议)DNS 协议的作用: 将域名转换为 IP 地址。 由于 HTTP 是基于 TCP 传输的、所以在发送 HTTP 请求前、需要进行三次握手、在客户端发送第一次握手的时候、( 浏览器向服务器发送一个SYN(同步)报文、其中包含客户端的初始序列号。TCP头部设置SYN标志位、并指定客户端端口 同时填上目标端口和源端口的信息。源端口是浏览器随机生成的、目标端口要看是 HTTP 还是 HTTPS、如果是 HTTP 默认目标端口是 80、如果是 HTTPS 默认是 443。(传输层) 然后到网络层:涉及到(IP协议) 会将TCP报文封装成IP数据包、添加IP头部,包含源IP地址(浏览器)和目标IP地址(服务器)。IP 协议的作用: 提供无连接的、不可靠的数据包传输服务。 然后到数据链路层、会通过 ARP 协议、获取目标的路由器的 MAC 地址、然后会加上 MAC 头、填上目标 MAC 地址和源 MAC 地址。 然后到物理层之后、直接把数据包、转发给路由器、路由器再通过下一跳、最终找到目标服务器、然后目标服务器收到客户的 SYN 报文后,会响应第二次握手。 当双方都完成三次握手后、如果是 HTTP 协议、客户端就会将 HTTP 请求就会发送给目标服务器。如果是 HTTPS 协议、客户端还要和服务端进行 TLS 四次握手之后、客户端才会将 HTTP 报文发送给目标服务器。 目标服务器收到 HTTP 请求消息后、就返回 HTTP 响应消息、浏览器会对响应消息进行解析渲染、呈现给用户
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客企业服务