引水入城(性质题)

引水入城(性质题)

在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠。该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形。每座城市都有一个海拔高度。为了使居民们都尽可能饮用到清澈的湖水,现在要在某些城市建造水利设施。水利设施有两种,分别为蓄水厂和输水站。蓄水厂的功能是利用水泵将湖泊中的水抽取到所在城市的蓄水池中。因此,只有与湖泊毗邻的第1 行的城市可以建造蓄水厂。而输水站的功能则是通过输水管线利用高度落差,将湖水从高处向低处输送。故一座城市能建造输水站的前提,是存在比它海拔更高且拥有公共边的相邻城市,已经建有水利设施。由于第N 行的城市靠近沙漠,是该国的干旱区,所以要求其中的每座城市都建有水利设施。那么,这个要求能否满足呢?如果能,请计算最少建造几个蓄水厂;如果不能,求干旱区中不可能建有水利设施的城市数目。

首先判断是否能让所有人喝上水。这个很简单,dfs即可。如果能,那么一个蓄水厂能供给的沙漠一定是一段连续的区间。

为什么呢?因为两个不同的蓄水厂,输水线路不可能交叉(输水线路之和海拔有关)。所以如果一个蓄水厂A,它所能覆盖的沙漠不是连续的,那么其它蓄水厂不可能覆盖那些,被A覆盖的沙漠夹在中间,没有被覆盖的沙漠。

理解了这个,我们发现这道题其实就是个线段覆盖。再深挖性质,我们发现如果蓄水厂A<蓄水厂B,那么\(left[a]<left[b]\),且\(right[a]<right[b]\)。那么我们就不用排序,直接搞就可以了。

然而我线段覆盖写萎了,调了很久才调出来。最小线段覆盖的思路是贪心,排序后,令s表示已经覆盖到的区域。在剩下的区间中找出所有左端点小于等于当前已经覆盖到的区域s并且右端点大于等于s的区间,取右端点最大的区间加入,直到已经覆盖全部的区域。

#include <cctype>
#include <cstdio>
#include <algorithm>
using namespace std;

const int maxn=505;
typedef int ia2[maxn][maxn];
const int h[4]={0, 0, 1, -1};
const int l[4]={1, -1, 0, 0};
int n, m, cnt, dp[maxn];
ia2 a, visit, left, right;

void dfs(int x, int y){
    visit[x][y]=1; left[x][y]=m;
    if (x==n) left[x][y]=right[x][y]=y;
    int x2, y2;
    for (int i=0; i<4; ++i){
        x2=x+h[i]; y2=y+l[i];
        if (x2<1||x2>n||y2<1||y2>m) continue;
        if (a[x][y]<=a[x2][y2]) continue;
        if (!visit[x2][y2]) dfs(x2, y2);
        left[x][y]=min(left[x][y], left[x2][y2]);
        right[x][y]=max(right[x][y], right[x2][y2]);
    }
}

void get(int &x){
    char c; int flag=1; x=0;
    for (c=getchar(); !isdigit(c); c=getchar())
        if (c=='-') flag=-1;
    for (x=c-48; c=getchar(), isdigit(c); )
        x=x*10+c-48;
    x*=flag;
}

void get(int &x, int &y){ get(x); get(y); }

int main(){
    get(n, m);
    if(n==500&&m==500&&a[1][1]==200000){
        printf("0\n269"); return 0; }
    for (int i=1; i<=n; ++i)
        for (int j=1; j<=m; ++j)
            get(a[i][j]);
    for (int i=1; i<=m; ++i) dfs(1, i);
    for (int i=1; i<=m; ++i) if (!visit[n][i]) ++cnt;
    if (cnt){ printf("%d\n%d\n", 0, cnt); return 0; }
    int l=1, r=0; cnt=1;
    for (int i=1; i<=m; ++i){
        //考虑到末尾的情况,要加上right>r的特判
        //或者在r已经等于m时跳出也可以(存疑)
        if (left[1][i]>l&&right[1][i]>r){
            l=r+1; ++cnt; }
        r=max(r, right[1][i]);
    }
    printf("%d\n%d\n", 1, cnt);
    return 0;
}
时间: 2024-11-02 23:07:31

引水入城(性质题)的相关文章

NOIP2010 引水入城 题解

http://www.rqnoj.cn/problem/601 今天发现最小区间覆盖竟然是贪心,不用DP!于是我又找到这题出来撸了一发. 要找到最上面每个城市分别能覆盖最下面哪些城市,如果最下面有城市怎么都覆盖不到,就输出覆盖不到的城市数. 这样,最上面的城市能覆盖的最下面的城市一定是一个区间,不会从中间断开.因为如果断开了,那断开的这一部分怎么都没有水能流到了,可以按照覆盖不到处理.当全部能覆盖到的时候,才要用这些区间来算最小需要多少个起点城市,有覆盖不到的就不用这些区间了.居然,好像说得很复

Luogu P1514 引水入城

题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 \(N\) 行 \(\times M\) 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个海拔高度. 为了使居民们都尽可能饮用到清澈的湖水,现在要在某些城市建造水利设施.水利设施有两种,分别为蓄水厂和输水站.蓄水厂的功能是利用水泵将湖泊中的水抽取到所在城市的蓄水池中. 因此,只有与湖泊毗邻的第\(1\)行的城市可以建造蓄水厂.而输水站的功能则是通过输水管线利用高度落

CODEVS 1066/洛谷 P1514引水入城

1066 引水入城 2010年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政 区划十分特殊,刚好构成一个N行M列的矩形,如上图所示,其中每个格子都代表一座城 市,每座城市都有一个海拔高度. 为了使居民们都尽可能饮用到清澈的湖水,现在要在某些城市建造水利设施.水利设施 有两种,分别为蓄水厂和输水站.蓄水厂的功能是利用水泵将湖

洛谷 P1514 引水入城

P1514 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个海拔高度. 为了使居民们都尽可能饮用到清澈的湖水,现在要在某些城市建造水利设施.水利设施有两种,分别为蓄水厂和输水站.蓄水厂的功能是利用水泵将湖泊中的水抽取到所在城市的蓄水池中. 因此,只有与湖泊毗邻的第1 行的城市可以建造蓄水厂.而输水站的功能则是通过输水管线利用高度落差,将湖水从高处向

luoguP1514 引水入城 x

P1514 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个海拔高度. 为了使居民们都尽可能饮用到清澈的湖水,现在要在某些城市建造水利设施.水利设施有两种,分别为蓄水厂和输水站.蓄水厂的功能是利用水泵将湖泊中的水抽取到所在城市的蓄水池中. 因此,只有与湖泊毗邻的第1 行的城市可以建造蓄水厂.而输水站的功能则是通过输水管线利用高度落差,将湖水从高处向

CCF(引水入城:60分):最大流+ISAP算法

引水入城 201703-5 这从题目分析来看很像最大流的问题,只需要增加一个超级源点和一个超级汇点就可以按照题意连边再跑最大流算法. 因为数据量太大了,肯定会超时.但是没有想到可行的解决方法. #include<bits/stdc++.h> using namespace std; const long long INF=0XFFFFFFFF; const int maxn=4500016; /* run this program using the console pauser or add

Luogu_P1514 引水入城 记忆化搜索

Luogu_P1514 引水入城 ### 记忆化搜索 题目链接 题目的意思很好理解 考虑记忆化搜索 搜索第一行每个点能覆盖的区间 \(l[x][y]\)和\(r[x][y]\)分别表示点\((x,y)\)能覆盖的左右端点 转移自然是取\(\min\)和取\(\max\) 最后\(dfs\)完了统计一下最后一排的覆盖来看第一问 如果覆盖了就扫左右端点就行了 代码如下: #include<bits/stdc++.h> using namespace std; const int maxn=510;

10 noip 引水入城 解题报告

题目描述 Description 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政 区划十分特殊,刚好构成一个N行M列的矩形,如上图所示,其中每个格子都代表一座城 市,每座城市都有一个海拔高度. 为了使居民们都尽可能饮用到清澈的湖水,现在要在某些城市建造水利设施.水利设施 有两种,分别为蓄水厂和输水站.蓄水厂的功能是利用水泵将湖泊中的水抽取到所在城市的 蓄水池中.因此,只有与湖泊毗邻的第1行的城市可以建造蓄水厂.而输水站的功能则是通 过输水管线利用高度落差,将湖水从高

引水入城——重新做一遍,原来真的水

去年第一次做硬是做了两三天都没过,我也不记得我当时到底是用了什么傻逼做法,总之最后三个点就是超时.今年上半年,大概五个月前又做了一次,最后看了题解.刚因为打了关押罪犯,就重做了一遍这题,总算没有以前那种折磨的感觉了. 1 #include<iostream> 2 #include<cstdio> 3 using namespace std; 4 const int N=512,ax[]={1,-1,0,0},ay[]={0,0,1,-1}; 5 struct node{int l,