【noip2010 洛谷p1514】引水入城

Before:

线段覆盖问题#1:(我们所需要的)

一个区间,若干条线段,现在求最少多少条线段覆盖满整个区间

区间长度8,可选的覆盖线段[2,6],[1,4],[3,6],[3,7],[6,8],[2,4],[3,5]

1将每一个区间按照左端点递增顺序排列,拍完序后为[1,4],[2,4],[2,6],[3,5],[3,6],[3,7],[6,8]

2设置一个变量表示已经覆盖到的区域。再剩下的线段中找出所有左端点小于等于当前已经覆盖到的区域的右端点的线段中,右端点最大的线段在加入,直到已经覆盖全部的区域

3过程:

假设第一步加入[1,4],那么下一步能够选择的有[2,6],[3,5],[3,6],[3,7],由于7最大,所以下一步选择[3,7],最后一步只能选择[6,8],这个时候刚好达到了8退出,所选区间为3

4贪心证明:

需要最少的线段进行覆盖,那么选取的线段必然要尽量长,而已经覆盖到的区域之前的地方已经无所谓了,(可以理解成所有的可以覆盖的左端点都是已经覆盖到的地方),那么真正能够使得线段更成的是右端点,左端点没有太大的意义,所以选择右端点来覆盖

线段覆盖问题#2:(我们所补充的)

一个区间,若干条线段,现在求最少删去多少条线段覆盖满整个区间

分析:

先将线段排序,先按线段起点排序,若起点相同按终点排序,都是由小到大,依次遍历线段判断是否保留。
设当前线段为[ai,bi],之前保留的线段的最右边的点为now,考虑以下情况:
1、ai>=now,即当前线段和之前保留的线段不重叠,那么直接保留当前线段(ans++),更新now=bi.
2、ai<now,当前线段和之前保留的线段有重叠,再分情况考虑:
(1)若bi<=now,证明当前线段完全被之前保留线段的区域包含(因为已经排序,所以当前线段的起点一定大于等于之前的线段,bi<now证明终点小于等于之前线段,所以被包含),那么证明当前线段更优(一条线段被另一条完全包含,显然取短的线段更优,即贪心),更新now=bi,但是结果计数(ans)不变.
(2)若bi>now,证明当前线段一部分和之前重叠,一部分不重叠,那么为了对后续线段影响最小,即now尽量小,用贪心法则,当前线段被舍弃,结果计数(ans)不变.

题目思路:

  1. 先bfs每一个沿湖城市所对应的沿沙漠城市。
  2. 沙漠城市如果有没有被bfs到的,说明有无法被覆盖到的点,直接进入无解状态;若全部都被bfs到了,进行线段覆盖,在沙漠里寻找最少的区间。

证明对于一个蓄水池,不可能出现流到沙漠地带时有间隔的情况,即:

不能出现从一个点i可以流到j-1和j+1却不能流到j的情况;

Why?

如果由i可以流到j-1和j+1却不能流到j,则j不可能被流到:

假设i流到j-1,j+1却没有流到j,那么h(j+1),h(j-1)<h(j),因此水无法从上下两边流到j,那么水只能从左边流到j。

假设有一点k可以到j

那么k到j的路线与i到j+1或j-1的路线一定有相交的地方,则通过i一定可以流到j,矛盾

即对于每一个沿湖城市,可以到达的沙漠城市都是一个区间。(线段覆盖问题)

讲一下毒瘤数据???

毒瘤数据#1:

n=1,只有一横行

看一下输入输出,这样的话会有几个点没有被加进已经被覆盖的数组中,定睛一看输入,我们发现爆0的地方都是区间最高点,当流到它时,接下来的地方就需要再建一个水库,因此未赋为1的点其实就是我们建水库的点。

加个特判:

毒瘤数据#2:

一个大到你无法想象的数据:

显然会t啊qwq?这时候显然需要优化啊。

一个简单的优化:在搜索第一行的每个点时,判断一下这个点的左边和这个点的右边,如果它比两边都高,说明没有点可以流到它,就bfs它,否则假设它左边或右边比它高,它一定可以看作是从左侧或右侧流过来的。

#include<bits/stdc++.h>

using namespace std;

int n,m,num,now,ans,_max;
int a[][];
bool vis[][],pan[];//vis每次bfs清空,表示某个点是否被搜索到过
//pan记录最后一行的点是否被覆盖到了 int dx[]={,,,-};
int dy[]={,-,,}; struct coder{
int l,r;
coder(){l=;r=;}//神奇的赋值构造函数?感谢sy?是在下才疏学浅了
}b[]; struct hyh{//用于bfs,记录横纵坐标
int x,y;
}; bool cmp(coder a,coder b){//按左端点从小到大排序
return a.l<b.l;
} hyh _hyh(int x,int y){hyh rtn;rtn.x=x;rtn.y=y;return rtn;}//赋值函数,与构造函数用途相同,但我不会写构造 bool check(){//判断最后一行是否都能被覆盖
int rtn=;
for(int i=;i<=m;i++)
if(!pan[i]){rtn=;num++;}
return rtn;
} bool zpd(int xx,int yy){//用于bfs时判断是否可以走
return vis[xx][yy]==&&xx>=&&yy>=&&xx<=n&&yy<=m;
} queue<hyh> q; void bfs(int x,const int f){
memset(vis,,sizeof(vis));
q.push(_hyh(,f));
vis[][f]=;
while(!q.empty()){
hyh c=q.front();
q.pop();
for(int i=;i<;i++){
int xx=c.x,yy=c.y;
xx+=dx[i];yy+=dy[i];
if(a[xx][yy]<a[c.x][c.y]&&zpd(xx,yy)){//满足从高流到低并且可以走
if(xx==n){//如果搜到的点是最后一行的
//记录第一行某个点能流到的最后一行的区间的左右端点分别是什么 :
if(b[f].l>yy) b[f].l=yy;
if(b[f].r<yy) b[f].r=yy;
pan[yy]=;//标记最后一行的某个点为可以走到
}
q.push(_hyh(xx,yy));
vis[xx][yy]=;
}
}
}
} int main(){
scanf("%d%d",&n,&m); for(int i=;i<=n;i++)
for(int j=;j<=m;j++)
scanf("%d",&a[i][j]); for(int i=;i<=m;i++)//优化:如果某个点左右两边的点都比这个点要低,bfs
if(a[][i]>=a[][i-]&&a[][i]>=a[][i+]) bfs(a[][i],i); if(!check()&&n!=){
//判断是否可以全部覆盖,如果不可以,直接进入无解状态(注意特殊的数据n==1)
printf("0\n");
printf("%d",num);
return ;
} if(n==){printf("1\n%d",num);return ;}//n==1的特判 sort(b+,b+m+,cmp);//进行线段覆盖,把线段按区间左端点位置排序; now=;_max=;//now我就认为是右端点了qwq,_max记录右端点的最大值;
while(now<=m){//当全部被覆盖,停止
for(int i=;i<=m;i++)
if(b[i].l<=now)//如果这个点左端点在当前覆盖区间之内
_max=max(_max,b[i].r); //找最大的右端点
ans++;//记录加了一个水库
now=_max+;//现在最大右端点更新为_max+1;(因为下一个线段的左端点可以刚好与之前的覆盖区间刚好相连):
//------ now=6,加入一个left=7,right=9的点 --------- (刚好相连)
}
printf("1\n");
printf("%d",ans);
return ;
}

最后插一句:why小姐姐说过了,无论何时,都要保持一个优雅的码风~~~(然而我的码风好像并不优雅

【五一qbxt】day7-1 引水入城的更多相关文章

  1. NOIP2010 引水入城

    4引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个 ...

  2. Codevs 1066 引水入城 2010年NOIP全国联赛提高组

    1066 引水入城 2010年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 传送门 题目描述 Description 在一个遥远的国度 ...

  3. CODEVS 1066/洛谷 P1514引水入城

    1066 引水入城 2010年NOIP全国联赛提高组  时间限制: 1 s  空间限制: 128000 KB  题目等级 : 钻石 Diamond   题目描述 Description 在一个遥远的国 ...

  4. Luogu 1514 引水入城 (搜索,动态规划)

    Luogu 1514 引水入城 (搜索,动态规划) Description 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N行M列的矩形,如上图 ...

  5. CCF CSP 201703-5 引水入城(50分)

    CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201703-5 引水入城 问题描述 MF城建立在一片高原上.由于城市唯一的水源是位于河谷地带的 ...

  6. 洛谷P1514 引水入城

    洛谷P1514 引水入城 原题链接 一道好题...细节真多 第一次提交90分,然后就GG了,不知从何改起 其实比较简单吧... 首先,一个点的水流向最后一排,一定可以形成一个区间. 不行的话肯定GG ...

  7. 洛谷 P1514 引水入城 解题报告

    P1514 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 NN 行 \times M×M 列的矩形,如上图所示,其中每个格 ...

  8. 洛谷P1514 引水入城 [搜索,区间DP]

    题目传送门 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 N 行×M 列的矩形,如上图所示,其中每个格子都代表一座城市,每 ...

  9. vijos p1777 引水入城(bfs+贪心)

    引水入城   描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N行M列的矩形,其中每个格子都代表一座城市,每座城市都有一个海拔高度. 为了使 ...

随机推荐

  1. leetcode69. x 的平方根 🌟

    题目: 实现 int sqrt(int x) 函数. 计算并返回 x 的平方根,其中 x 是非负整数. 由于返回类型是整数,结果只保留整数的部分,小数部分将被舍去. 示例 1: 输入: 4 输出: 2 ...

  2. 动态规划—distinct-subsequences

    题目: Given a string S and a string T, count the number of distinct subsequences of T in S. A subseque ...

  3. [Tyvj1423]GF和猫咪的玩具(最短路)

    [Tyvj1423]GF和猫咪的玩具 题目描述 GF同学和猫咪得到了一个特别的玩具,这个玩具由n个金属环(编号为1---n),和m条绳索组成,每条绳索连接两个不同的金属环,并且长度相同.GF左手拿起金 ...

  4. maven 提取jar包 依赖及打包排除

    <properties> <project.targetDir>D:\jar</project.targetDir> <project.targetServe ...

  5. mysql错误: waiting for table metadata lock

    今天突然发现truncate一个表都慢到不行,于是 SHOW PROCESSLIST 发现错误:waiting for table metadata lock解决方法:查看information_sc ...

  6. super语句不必须放在方法第一行。

    class A(object): pass class B(A): def __init__(self): self.__a = "B#a" super(B, self).__in ...

  7. oracle 11g 执行先决条件检查失败的解决方法

    在安装oracle 11g时,出现执行先决条件失败的情况如下: 你可以忽略所有强制安装,一般不会影响功能,但如果你想知道为什么会产生这种错误, 并且当出现以上情况时又该如何解决呢?如下列出了原因和解决 ...

  8. 转载-使用Nodepad++来编辑我们服务器的配置文件

    转自------------------ 作者:李阿昀 来源:CSDN 原文:https://blog.csdn.net/yerenyuan_pku/article/details/73128819 ...

  9. OC + RAC (九) 过滤

    // 跳跃 : 如下,skip传入2 跳过前面两个值 // 实际用处: 在实际开发中比如 后台返回的数据前面几个没用,我们想跳跃过去,便可以用skip - (void)skip { RACSubjec ...

  10. Sklearn----使用决策树预测隐形眼镜类型

    import pandas as pd import pydotplus from sklearn.externals.six import StringIO #LabelEncoder:将字符串转换 ...