题解【洛谷P1514】[NOIP2010]引水入城
题目描述
在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠。该国的行政区划十分特殊,刚好构成一个 \(N\) 行 \(M\) 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个海拔高度。
为了使居民们都尽可能饮用到清澈的湖水,现在要在某些城市建造水利设施。水利设施有两种,分别为蓄水厂和输水站。蓄水厂的功能是利用水泵将湖泊中的水抽取到所在城市的蓄水池中。因此,只有与湖泊毗邻的第 \(1\) 行的城市可以建造蓄水厂。而输水站的功能则是通过输水管线利用高度落差,将湖水从高处向低处输送。故一座城市能建造输水站的前提,是存在比它海拔更高且拥有公共边的相邻城市,已经建有水利设施。
由于第 \(N\) 行的城市靠近沙漠,是该国的干旱区,所以要求其中的每座城市都建有水利设施。那么,这个要求能否满足呢?如果能,请计算少建造几个蓄水厂;如果不能,求干旱区中不可能建有水利设施的城市数目。
输入格式
输入的每行中两个数之间用一个空格隔开。
输入的第一行是两个正整数 \(N\) 和 \(M\),表示矩形的规模;
接下来 \(N\) 行,每行 \(M\) 个正整数,依次代表每座城市的海拔高度。
输出格式
输出有两行。
如果能满足要求,输出的第一行是整数 \(1\) ,第二行是一个整数,代表最少建造几个蓄水厂;
如果不能满足要求,输出的第一行是整数 \(0\) ,第二行是一个整数,代表有几座干旱区中的城市不可能建有水利设施。
样例
样例输入 1
2 5
9 1 5 4 3
8 7 6 1 2
样例输出 1
1
1
样例说明 1
只需要在海拔为 \(9\) 的那座城市中建造蓄水厂,即可满足要求。
样例输入 2
3 6
8 4 5 6 4 4
7 3 4 3 3 3
3 2 2 1 1 2
样例输出 2
1
3
样例说明 2
上图中,在 \(3\) 个粗线框出的城市中建造蓄水厂,可以满足要求。以这 \(3\) 个蓄水厂为源头在干旱区中建造的输水站分别用 \(3\) 种颜色标出。当然,建造方法可能不唯一。
数据范围与提示
题解
考虑枚举第一行的每个位置进行\(\mathrm{DFS}\)扩展,标记已经能够覆盖到的点。
不难发现每一个点覆盖的区间都是连续的,证明留给读者作为练习。
如果一个点没有被覆盖到,说明它的海拔比它的相邻城市高。
记录每个点能够覆盖到的最左边的点\(\mathrm{l[i][j]}\)和最右边的点\(\mathrm{r[i][j]}\),分别表示第\(i\)行第\(j\)列能够覆盖到的最左端点和最右端点。
\(\mathrm{DFS}\)的过程中更新\(\mathrm{l[i][j]}\)和\(\mathrm{r[i][j]}\)即可。
最后统计答案并输出。
具体实现细节参考代码注释。
代码
/********************************
Author: csxsl
Date: 2019/10/28
Language: C++
Problem: P1514
********************************/
#include <bits/stdc++.h>
#define itn int
#define gI gi
using namespace std;
inline int gi()
{
int f = 1, x = 0; char c = getchar();
while (c < '0' || c > '9') {if (c == '-') f = -1; c = getchar();}
while (c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return f * x;
}
inline long long gl()
{
long long f = 1, x = 0; char c = getchar();
while (c < '0' || c > '9') {if (c == '-') f = -1; c = getchar();}
while (c >= '0' && c <= '9') x = x * 10 + c - '0', c = getchar();
return f * x;
}
const int maxn = 503;
int n, m, cnt, l[maxn][maxn], r[maxn][maxn], ans, a[maxn][maxn], vis[maxn][maxn];
const int dx[4] = {-1, 1, 0, 0}, dy[4] = {0, 0, 1, -1};//定义方向数组
void dfs(int x, int y)
{
vis[x][y] = 1;//标记访问过的点,即能够覆盖到的点
for (int i = 0; i < 4; i+=1)
{
int xx = x + dx[i], yy = y + dy[i];//向前后左右扩展
if (xx < 1 || xx > n || yy < 1 || yy > m) continue;//判断边界
if (a[xx][yy] >= a[x][y]) continue;//不能流
if (!vis[xx][yy]) dfs(xx, yy);//进行递归
l[x][y] = min(l[x][y], l[xx][yy]);//更新l[i][j]
r[x][y] = max(r[x][y], r[xx][yy]);//更新r[i][j]
}
}
int main()
{
n = gi(), m = gi();
memset(l, 0x3f, sizeof(l));//初始化极大值
for (int i = 1; i <= m; i+=1) l[n][i] = r[n][i] = i;//第n行上的每个点最少能覆盖自己
for (int i = 1; i <= n; i+=1)
{
for (int j = 1; j <= m; j+=1)
{
a[i][j] = gi();//输入海拔高度
}
}
for (int i = 1; i <= m; i+=1) if (!vis[1][i]) dfs(1, i);//进行扩展
bool fl = true;
for (int i = 1; i <= m; i+=1) if (!vis[n][i]) {fl = false; break;}//第n行还有没被覆盖的点
if (!fl) //不能满足要求
{
puts("0");
for (int i = 1; i <= m; i+=1) if (!vis[n][i]) ++cnt;//记录不能覆盖的城市个数
printf("%d\n", cnt);//输出
return 0;
}
puts("1");//可以满足要求
int maxleft = 1;
while (maxleft <= m)//开始向右覆盖
{
int maxright = 0;
for (int i = 1; i <= m; i+=1)
{
if (l[1][i] <= maxleft) maxright = max(maxright, r[1][i]);//可以覆盖当前区间
}
maxleft = maxright + 1;//更新新的左端点
++cnt;//增加答案
}
printf("%d\n", cnt);//输出
return 0;//结束
}
题解【洛谷P1514】[NOIP2010]引水入城的更多相关文章
- [洛谷P1514] NOIP2010 引水入城
问题描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行×M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个海拔高度. ...
- 【题解】洛谷P1514 [NOIP2010TG] 引水入城(DFS+DP)
次元传送门:洛谷P1514 思路 可以证明如果有解 那么每个蓄水池可以覆盖到的干旱区必定是线段 证明: 举个栗子 8 9 8 7 9 7 6 9 6 明显到不了中间的点 如果不是连续的线段 中间肯定有 ...
- 洛谷 P1514 【引水入城】
题库 :洛谷 题号 :1514 题目 :引水入城 link :https://www.luogu.org/problemnew/show/P1514 思路 :搜索从第一排开始能覆盖最后一排的区间L ~ ...
- 洛谷 1514 (NOIp2010) 引水入城
题目:https://www.luogu.org/problemnew/show/P1514 如果有解,一个第一行的格子能覆盖第n行的一定是一个连续的区间. 因为如果不连续,则有围住了一些第n行的格子 ...
- 洛谷 P 1514 引水入城==Codevs 1066
题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个海拔高度. ...
- 521. [NOIP2010] 引水入城 cogs
521. [NOIP2010] 引水入城 ★★★ 输入文件:flow.in 输出文件:flow.out 简单对比时间限制:1 s 内存限制:128 MB 在一个遥远的国度,一侧是风景秀 ...
- 洛谷P1514 [NOIP2010提高组T4]引水入城
P1514 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城 ...
- NOIP2010 引水入城
4引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个 ...
- NOIP2010 引水入城 题解
http://www.rqnoj.cn/problem/601 今天发现最小区间覆盖竟然是贪心,不用DP!于是我又找到这题出来撸了一发. 要找到最上面每个城市分别能覆盖最下面哪些城市,如果最下面有城市 ...
随机推荐
- CF1300E Water Balance
题目链接 problem 给出一个长度为n的序列,每次可以选择一个区间\([l,r]\)并将区间\([l,r]\)内的数字全部变为这些数字的平均数.该操作可以进行任意多次. 求出进行任意次操作后可以得 ...
- 【BZOJ 1022】 [SHOI2008]小约翰的游戏John(Anti_SG)
Description 小约翰经常和他的哥哥玩一个非常有趣的游戏:桌子上有n堆石子,小约翰和他的哥哥轮流取石子,每个人取 的时候,可以随意选择一堆石子,在这堆石子中取走任意多的石子,但不能一粒石子也不 ...
- 使用IntersectionObserver制作滚动动画以及其他记录
前言 最近在重做公司项目的主页,正好新来了个UI,整个都重新设计了一下,动画还挺多的.我之前没有怎么玩过这些,踩了挺多坑,最后找到了目前而言最合适的方法,现在做一个记录. 需要把原来的主页从项目中抽出 ...
- 从首个IMO季军谈起 作者 : 付云皓
刚刚过去的IMO,中国史无前例地获得了第三名,也是自1997年来近20年首次跌出前二.感谢微信等社交软件,相信现在这个新闻已经以火箭的速度传播了. 作为一个与数学竞赛及IMO打了多年交道的人,我一直有 ...
- BZOJ4566&&lg3181 HAOI找相同字符(广义后缀自动机)
BZOJ4566&&lg3181 HAOI找相同字符(广义后缀自动机) 题面 自己找去 HINT 给定两个文本串,问从两个串中各取一个非空子串,使这俩子串相同,问方案有多少种.我的思路 ...
- git三剑客笔记
看了git三剑客视频总结的笔记,只给自己参考. 常用命令 查看分支:git branch 创建分支:git branch <name> 切换分支:git checkout <name ...
- Ubuntu 打不开终端 侧边栏消失的解决办法
在网上找了很多办法,大多不行,具体原因也不太清楚,应该是Unity某些配置被改了. 我是在ubuntu14.04平台利用apt-get卸载python后,关机重启出现"打不开终端和侧边栏消失 ...
- 文本harry potter的字符统计
实现计算文件中字符的占比和不同单词的个数两项功能,首先将文本文件按行导入到程序中,再通过charAT()函数来实现对单个字符的操作,并用集合来统计字符总数以及不同的字符的个数,进而输出各个字符的个数以 ...
- 【资源分享】Gmod自瞄自动开枪脚本
*----------------------------------------------[下载区]----------------------------------------------* ...
- 【Python】表白代码
# -*- coding:utf-8 -*- import turtle import time # 画爱心的顶部 def LittleHeart(): for i in range(200): tu ...