NOI2016区间bzoj4653(线段树,尺取法,区间离散化)
题目描述
在数轴上有 \(N\) 个闭区间 \([l_1,r_1],[l_2,r_2],...,[l_n,r_n]\) 。现在要从中选出 \(M\) 个区间,使得这 \(M\) 个区间共同包含至少一个位置。换句话说,就是使得存在一个 \(x\) ,使得对于每一个被选中的区间 \([l_i,r_i]\) ,都有 \(l_i≤x≤r_i\) 。
对于一个合法的选取方案,它的花费为被选中的最长区间长度减去被选中的最短区间长度。区间 \([l_i,r_i]\) 的长度定义为 \(r_i-l_i\) ,即等于它的右端点的值减去左端点的值。
求所有合法方案中最小的花费。如果不存在合法的方案,输出 \(-1\) 。
输入输出格式
输入格式:
第一行包含两个正整数 \(N,M\) 用空格隔开,意义如上文所述。保证 \(1≤M≤N\)
接下来 \(N\) 行,每行表示一个区间,包含用空格隔开的两个整数 \(l_i\) 和 \(r_i\) 为该区间的左右端点。
\(N<=500000,M<=200000,0≤li≤ri≤10^9\)
输出格式:
只有一行,包含一个正整数,即最小花费。
首先,这个区间的数值非常的大,我们需要离散化
可是我QwQ不会呀
区间离散化是将左端点,和右端点放到同一个数组里,排序,然后依次赋值。
记得区间长度计算要在离散化之前
void init()
{
for (int i=1;i<=n;i++)
{
b[++cnt].val=a[i].l;
b[cnt].bel=1;b[cnt].num=i;
b[++cnt].val=a[i].r;
b[cnt].bel=2;b[cnt].num=i;
}
sort(b+1,b+1+cnt,cmp);
for (int i=1;i<=cnt;i++)
{
if (b[i].val!=b[i-1].val) ymh++;
if (b[i].bel==1) l[b[i].num]=ymh;
else r[b[i].num]=ymh;
}
for (int i=1;i<=n;i++)
{
a[i].l=l[i];
a[i].r=r[i];
}
//for (int i=1;i<=n;i++) printf("%d %d\n",a[i].l,a[i].r);
sort(a+1,a+1+n,cmp1);
}
离散化初始化完之后呢~
我们考虑这个题是要求使得最大区间减去最小区间的值最小,那么我们不妨按照区间长度排序
那么,如果判断一个点是否被覆盖了m次呢。
我们可以直接对于每个\([l,r]\)将里面的数都加一,然后统计区间最大值就可以了
那么就需要一个线段树了!
那.....之后呢QwQ,好像没什么用。
这时候就需要一个神奇的东西“尺取法”
尺取法:顾名思义,像尺子一样取一段,借用挑战书上面的话说,尺取法通常是对数组保存一对下标,即所选取的区间的左右端点,然后根据实际情况不断地推进区间左右端点以得出答案。之所以需要掌握这个技巧,是因为尺取法比直接暴力枚举区间效率高很多,尤其是数据量大的
时候,所以尺取法是一种高效的枚举区间的方法,一般用于求取有一定限制的区间个数或最短的区间等等。当然任何技巧都存在其不足的地方,有些情况下尺取法不可行,无法得出正确答案
尺取法通常适用于选取区间有一定规律,或者说所选取的区间有一定的变化趋势的情况,通俗地说,在对所选取区间进行判断之后,我们可以明确如何进一步有方向地推进区间端点以求解满足条件的区间,如果已经判断了目前所选取的区间,但却无法确定所要求解的区间如何进一步
得到根据其端点得到,那么尺取法便是不可行的。首先,明确题目所需要求解的量之后,区间左右端点一般从最整个数组的起点开始,之后判断区间是否符合条件在根据实际情况变化区间的端点求解答案。
举个例子:
我们要求在给定的序列中求和等于x的区间的个数
那么我们对于当前区间\([l,r]\),如果当前的和小于x,就\(r++\) ,否则\(l++\)
回到这道题
我们先按照长度从小到大依次加入区间,如果当然已经存在一个出现次数为\(m\)
那么就更新答案,并跳\(l\),如果已知出现次数是m,那么就一直更新答案了
否则就一直跳r
QwQ不过跳的时候边界和更新与跳指针的先后要特别注意,详细直接看代码吧
#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<map>
#include<queue>
#include<vector>
#include<ctime>
using namespace std;
inline int read()
{
int x=0,f=1;char ch=getchar();
while (!isdigit(ch)){if (ch=='-') f=-1;ch=getchar();}
while (isdigit(ch)){x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
return x*f;
}
const int maxn = 500010;
struct Node{
int l,r,len;
};
struct pp{
int val,num,bel;
};
Node a[maxn];
pp b[2*maxn];
int n,m;
int f[10*maxn];
int cnt;
int st,ed;
int add[10*maxn];
int l[maxn],r[maxn];
bool cmp(pp a,pp b)
{
return a.val<b.val;
}
bool cmp1(Node a,Node b)
{
return a.len<b.len;
}
int ymh=0;
void init()
{
for (int i=1;i<=n;i++)
{
b[++cnt].val=a[i].l;
b[cnt].bel=1;b[cnt].num=i;
b[++cnt].val=a[i].r;
b[cnt].bel=2;b[cnt].num=i;
}
sort(b+1,b+1+cnt,cmp);
for (int i=1;i<=cnt;i++)
{
if (b[i].val!=b[i-1].val) ymh++;
if (b[i].bel==1) l[b[i].num]=ymh;
else r[b[i].num]=ymh;
}
for (int i=1;i<=n;i++)
{
a[i].l=l[i];
a[i].r=r[i];
}
//for (int i=1;i<=n;i++) printf("%d %d\n",a[i].l,a[i].r);
sort(a+1,a+1+n,cmp1);
}
void up(int root)
{
f[root]=max(f[root<<1],f[root<<1|1]);
}
void pushdown(int root,int l,int r)
{
if (add[root])
{
add[root<<1]+=add[root];
add[root<<1|1]+=add[root];
f[root<<1]=f[root<<1]+add[root];
f[root<<1|1]=f[root<<1|1]+add[root];
add[root]=0;
}
}
void update(int root,int l,int r,int x,int y,int p)
{
if (x<=l && r<=y)
{
f[root]=f[root]+p;
add[root]+=p;
return;
}
pushdown(root,l,r);
int mid =(l+r) >> 1;
if (x<=mid) update(root<<1,l,mid,x,y,p);
if (y>mid) update(root<<1|1,mid+1,r,x,y,p);
up(root);
}
int query(int root,int l,int r,int x,int y)
{
if (x<=l && r<=y)
{
return f[root];
}
pushdown(root,l,r);
int mid = (l+r) >> 1;
int ans=0;
if (x<=mid) ans=max(ans,query(root<<1,l,mid,x,y));
if (y>mid) ans=max(ans,query(root<<1|1,mid+1,r,x,y));
return ans;
}
int ans=2e9;
int main()
{
n=read(),m=read();
for (int i=1;i<=n;i++)
{
a[i].l=read(),a[i].r=read();
a[i].len=a[i].r-a[i].l;
}
init();
st=1;ed=1;
update(1,1,ymh,a[1].l,a[1].r,1);
while (ed<n && st<n)
{
while (query(1,1,ymh,1,ymh)<m && ed<n && st<n)
{
++ed;
update(1,1,ymh,a[ed].l,a[ed].r,1);
}
while(query(1,1,ymh,1,ymh)>=m && ed<n && st<n && st<ed)
{
if (query(1,1,ymh,1,ymh)==m)
ans=min(ans,a[ed].len-a[st].len);
update(1,1,ymh,a[st].l,a[st].r,-1);
++st;
}
}
if (ans==2e9) ans=-1;
cout<<ans<<endl;
return 0;
}
NOI2016区间bzoj4653(线段树,尺取法,区间离散化)的更多相关文章
- 2018.08.17 bzoj4653: [Noi2016]区间(线段树+尺取法)
传送门 将坐标离散化之后直接用尺取法(双指针)+线段树维护. 其实就是说只要目前所有点的被覆盖次数是大于等于m的就移动左指针删除区间更新答案,否则移动右指针加入区间更新答案. 话说忘记排序以及建树的时 ...
- P1712-[NOI2016]区间【线段树,尺取法】
正题 题目链接:https://www.luogu.com.cn/problem/P1712 题目大意 \(n\)个区间,求出其中\(m\)个区间使得它们有覆盖同一个点且最长区间长度减去最短长度最小. ...
- 【NOI2016】区间 题解(线段树+尺取法)
题目链接 题目大意:给定$n$个区间$[l_i,r_i]$,选出$m$个区间使它们有一个共同的位置$x$,且使它们产生的费用最小.求最小费用.费用定义为最长的区间长度减去最短区间长度. ------- ...
- luogu 1712 区间(线段树+尺取法)
题意:给出n个区间,求选择一些区间,使得一个点被覆盖的次数超过m次,最小的花费.花费指的是选择的区间中最大长度减去最小长度. 坐标值这么大,n比较小,显然需要离散化,需要一个技巧,把区间转化为半开半闭 ...
- 【洛谷 P1712】 [NOI2016]区间 (线段树+尺取)
题目链接 emmm看起来好像无从下手, \(l_i,r_i\)这么大,肯定是要离散化的. 然后我们是选\(m\)个区间,我们先对这些区间按长度排个序也不影响. 排序后,设我们取的\(m\)个区间的编号 ...
- 【BZOJ4653】【NOI2016】区间(线段树)
[BZOJ4653][NOI2016]区间(线段树) 题面 BZOJ 题解 \(NOI\)良心送分题?? 既然是最大长度减去最小长度 莫名想到那道反复减边求最小生成树 从而求出最小的比值 所以这题的套 ...
- [BZOJ4653][NOI2016]区间 贪心+线段树
4653: [Noi2016]区间 Time Limit: 60 Sec Memory Limit: 256 MB Description 在数轴上有 n个闭区间 [l1,r1],[l2,r2],. ...
- 【题解】P1712 [NOI2016]区间(贪心+线段树)
[题解]P1712 [NOI2016]区间(贪心+线段树) 一个observe是,对于一个合法的方案,将其线段长度按照从大到小排序后,他极差的来源是第一个和最后一个.或者说,读入的线段按照长度分类后, ...
- BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针
BZOJ_4653_[Noi2016]区间_线段树+离散化+双指针 Description 在数轴上有 n个闭区间 [l1,r1],[l2,r2],...,[ln,rn].现在要从中选出 m 个区间, ...
- HDU 1754 I Hate It(线段树单点替换+区间最值)
I Hate It [题目链接]I Hate It [题目类型]线段树单点替换+区间最值 &题意: 本题目包含多组测试,请处理到文件结束. 在每个测试的第一行,有两个正整数 N 和 M ( 0 ...
随机推荐
- JAVAWEB开发批量删除,SSM的几种情况
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Ubuntu 16.04 NVidia显卡 输入密码后 重复出现登录界面
问题根源:显卡驱动 解决办法: CTRL+ALT+F1 # 切换到命令行 sudo service lightdm stop # 关闭桌面显示管理器 sudo apt-get remove --pu ...
- Linux系统的日志管理、时间同步、延迟命令at
方便查看和管理 /var/log/messages ?系统服务及日志,包括服务的信息,报错等等 /var/log/secure ? ? ? ? 系统认证信息日志 /var/log/maillog ? ...
- Java 学习:数据类型
前言:Java属于强类型语言 强类型语言:要求变量的使用要严格符合规定,所有变量都必须先定义后才能使用 优势就是安全性高,但劣势速度慢 数据类型 Java的数据类型分为两大类: 基本类型(primit ...
- Java基础之SPI机制
SPI 机制,全称为 Service Provider Interface,是一种服务发现机制.它通过在 ClassPath 路径下的 META-INF/services 文件夹查找文件,自动加载文件 ...
- Appium问题解决方案(10)- Original error: Swipe did not complete successfully
背景 从搜索页面返回首页之后,执行 swipe 滑动操作,但是报错了,如上图 解决方法 只需要在第一次 swipe 之前加个 sleep,强制等待即可 备注 这种解决方案其实不好,强制等待能少用就少 ...
- openswan协商流程之(二):main_inI1_outR1()
主模式第二包:main_inI1_outR1() 文章目录 主模式第二包:main_inI1_outR1() 1. 序言 2. `main_inI1_outR1()`处理流程图 3. `main_in ...
- GridView控件使用
增加显示列gridView.Columns.AddVisible("AgentName", "姓名");设置是否为只读gridView1.OptionsBeha ...
- WinUI 3学习笔记(3)—— ComboBox & DropDownButton & SplitButton
本篇想介绍相对小众但颇具使用价值的控件SplitButton,提到SplitButton难免会拿来与ComboBox进行比较,同时在WinUI 3的控件库中,还有一个默默无闻的DropDownButt ...
- prometheus从零开始
本次的想法是做服务监控 并告警 主要线路如下图所示 1.运行prometheus docker方式 docker run -itd \ -p 9090:9090 \ -v /opt/prometh ...