【五一qbxt】day7-1 引水入城
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)不变.
题目思路:
- 先bfs每一个沿湖城市所对应的沿沙漠城市。
- 沙漠城市如果有没有被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 引水入城的更多相关文章
- NOIP2010 引水入城
4引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N 行M 列的矩形,如上图所示,其中每个格子都代表一座城市,每座城市都有一个 ...
- Codevs 1066 引水入城 2010年NOIP全国联赛提高组
1066 引水入城 2010年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 传送门 题目描述 Description 在一个遥远的国度 ...
- CODEVS 1066/洛谷 P1514引水入城
1066 引水入城 2010年NOIP全国联赛提高组 时间限制: 1 s 空间限制: 128000 KB 题目等级 : 钻石 Diamond 题目描述 Description 在一个遥远的国 ...
- Luogu 1514 引水入城 (搜索,动态规划)
Luogu 1514 引水入城 (搜索,动态规划) Description 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N行M列的矩形,如上图 ...
- CCF CSP 201703-5 引水入城(50分)
CCF计算机职业资格认证考试题解系列文章为meelo原创,请务必以链接形式注明本文地址 CCF CSP 201703-5 引水入城 问题描述 MF城建立在一片高原上.由于城市唯一的水源是位于河谷地带的 ...
- 洛谷P1514 引水入城
洛谷P1514 引水入城 原题链接 一道好题...细节真多 第一次提交90分,然后就GG了,不知从何改起 其实比较简单吧... 首先,一个点的水流向最后一排,一定可以形成一个区间. 不行的话肯定GG ...
- 洛谷 P1514 引水入城 解题报告
P1514 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 NN 行 \times M×M 列的矩形,如上图所示,其中每个格 ...
- 洛谷P1514 引水入城 [搜索,区间DP]
题目传送门 引水入城 题目描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个 N 行×M 列的矩形,如上图所示,其中每个格子都代表一座城市,每 ...
- vijos p1777 引水入城(bfs+贪心)
引水入城 描述 在一个遥远的国度,一侧是风景秀美的湖泊,另一侧则是漫无边际的沙漠.该国的行政区划十分特殊,刚好构成一个N行M列的矩形,其中每个格子都代表一座城市,每座城市都有一个海拔高度. 为了使 ...
随机推荐
- 一、bootstrap-upload
一.bootstrap-upload 前端代码: @{ Layout = null; } <!DOCTYPE html> <html lang="zh-CN"&g ...
- 重启uwsgi
sudo pkill -f uwsgi -9 uwsgi --ini mysite.uwsgi.ini
- jmeter中遇见的坑:url需要编码的
在postman中能请求成功,但是在jmeter就是请求失败报500错. 请求的 url :/graph/vertices?label=node&properties={"num& ...
- python数据分析第二版:数据加载,存储和格式
一:读取数据的函数 1.读取csv文件 import numpy as np import pandas as pd data = pd.read_csv("C:\\Users\\Admin ...
- pandas.DataFrame.drop_duplicates 用法说明
DataFrame.drop_duplicates(subset=None, keep='first', inplace=False) subset考虑重复发生在哪一列,默认考虑所有列,就是在任何一列 ...
- 在一个div上增加遮罩
有一个需求是给一个视频增加遮罩 我研究了下 抽象出来就是给一个div增加遮罩 原理是:最外层的div使用relative定位 然后里面放两个子div 一个是不被遮的 另一个是遮罩(用abs ...
- securityDemopom
<properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> &l ...
- Django2 + ORM
创建模型类class UserInfo(models.Model): id = models.IntegerField() username = models.CharField(max_length ...
- Xcode模拟器快捷键
command + 左右 = 横竖屏旋转 command + H + H = 切入层级后台模式
- 打印XX提交的svn版本信息
打印出匹配uliuchao或--结尾的行 svn log | sed -n '/uliuchao/,/--$/p'