NOI2009 区间
题目链接:戳我
60分部分分还是很好拿的,排序(按照左端点为第一关键字,右端点为第二关键字)之后一个\(O(n^2)\),暴力判交,更新最小值,就可以水过前12个测试点。
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
#define MAXN 100010
#define inf 0x3f3f3f3f
using namespace std;
int n,m,ans=inf;
struct Node{int l,r;}node[MAXN];
inline bool cmp(struct Node x,struct Node y)
{
if(x.l==y.l) return x.r<y.r;
return x.l<y.l;
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("ce.in","r",stdin);
#endif
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
{
scanf("%d%d",&node[i].l,&node[i].r);
if(node[i].l==node[i].r) i--,n--;
}
sort(&node[1],&node[n+1],cmp);
for(int i=1;i<=n;i++)
{
vector<int>v;
int cnt=1;
v.push_back(node[i].r-node[i].l);
for(int j=1;j<=n;j++)
{
if(node[j].r<node[i].r||j==i||node[j].l>=node[i].r) continue;
v.push_back(node[j].r-node[j].l);
cnt++;
}
if(cnt<m) continue;
sort(v.begin(),v.end());
for(int i=0;i<v.size();i++)
{
int j=i+m-1;
if(j>=v.size()) break;
ans=min(ans,v[j]-v[i]);
}
}
if(ans==inf) printf("-1\n");
else printf("%d\n",ans);
return 0;
}
现在我们来考虑正解——
区间1e9有点大,先离散化一下。
然后我们按照长度从小到大排序qwq,保证一个单调性,然后使用尺取法。给当前枚举到的线段所指区间+1,用线段树维护全局最大值,如果最大值==m,就可以用右指针所指线段的长度-左指针所指线段的长度更新答案了。如果当前全局最大值大于m,我们可以弹出原先的线段。
什么是尺取法?就是维护两个指针一样的东西( l,r) ,每次确定区间的左端点,让\(r\)不断向右移动,直到满足条件停下,维护一下答案,直到\(r>n\) ]
不过,注意一下,不要把node[i].l==node[i].r,省略掉qwqwq(不然会WA在UOJ上)
代码如下:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
#define MAXN 500010
#define INF 2147483647
using namespace std;
int n,m,ans=INF,cnt;
int pre[MAXN<<1];
struct Node{int maxx,tag;}t[MAXN<<4];
struct Node2{int l,r,len,id;}node[MAXN<<1];
inline int ls(int x){return x<<1;}
inline int rs(int x){return x<<1|1;}
inline void push_up(int x){t[x].maxx=max(t[ls(x)].maxx,t[rs(x)].maxx);}
inline bool cmp(struct Node2 x,struct Node2 y){return x.len<y.len;}
inline void solve(int x,int k)
{
t[x].maxx+=k;
t[x].tag+=k;
}
inline void push_down(int x)
{
if(t[x].tag)
{
solve(ls(x),t[x].tag);
solve(rs(x),t[x].tag);
t[x].tag=0;
}
}
inline void update(int x,int l,int r,int ll,int rr,int k)
{
if(ll<=l&&r<=rr) {solve(x,k);return;}
int mid=(l+r)>>1;
push_down(x);
if(ll<=mid) update(ls(x),l,mid,ll,rr,k);
if(mid<rr) update(rs(x),mid+1,r,ll,rr,k);
push_up(x);
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("ce.in","r",stdin);
#endif
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++)
{
scanf("%d%d",&node[i].l,&node[i].r);
node[i].len=node[i].r-node[i].l;
node[i].id=i;
}
for(int i=1;i<=n;i++) pre[++cnt]=node[i].l,pre[++cnt]=node[i].r;
sort(&pre[1],&pre[cnt+1]);
cnt=unique(&pre[1],&pre[1+cnt])-pre-1;
for(int i=1;i<=n;i++)
{
node[i].l=lower_bound(&pre[1],&pre[cnt+1],node[i].l)-pre;
node[i].r=lower_bound(&pre[1],&pre[cnt+1],node[i].r)-pre;
}
sort(&node[1],&node[1+n],cmp);
int head=0,tail=0;
for(;;)
{
while(tail<n&&t[1].maxx<m)
{
++tail;
update(1,1,cnt,node[tail].l,node[tail].r,1);
}
if(t[1].maxx<m) break;
while(head<n&&t[1].maxx>=m)
{
++head;
update(1,1,cnt,node[head].l,node[head].r,-1);
}
ans=min(ans,node[tail].len-node[head].len);
}
if(ans==INF) printf("-1\n");
else printf("%d\n",ans);
return 0;
}
注意这种离散化的题目!!!线段树要开8倍qwqwq
要不然会RE掉三个点qwqwq
NOI2009 区间的更多相关文章
- NOI2009 二叉查找树 【区间dp】
[NOI2009]二叉查找树 [问题描述] 已知一棵特殊的二叉查找树.根据定义,该二叉查找树中每个结点的数据值都比它左子树结点的数据值大,而比它右子树结点的数据值小.另一方面,这棵查找树中每个结点都有 ...
- bzoj 1564 [NOI2009]二叉查找树 区间DP
[NOI2009]二叉查找树 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 906 Solved: 630[Submit][Status][Discu ...
- BZOJ1564 NOI2009二叉查找树(区间dp)
首先按数据值排序,那么连续一段区间的dfs序一定也是连续的. 将权值离散化,设f[i][j][k]为i到j区间内所有点的权值都>=k的最小代价,转移时枚举根考虑是否修改权值即可. #includ ...
- [BZOJ1564][NOI2009]二叉查找树 树形dp 区间dp
1564: [NOI2009]二叉查找树 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 879 Solved: 612[Submit][Status] ...
- 洛谷$P1864\ [NOI2009]$二叉查找树 区间$dp$
正解:区间$dp$ 解题报告: 传送门$QwQ$ 首先根据二叉查找树的定义可知,数据确定了,这棵树的中序遍历就已经改变了,唯一能改变的就是通过改变权值从而改变结点的深度. 发现这里权值的值没有意义,所 ...
- [BZOJ1564][NOI2009]二叉查找树(区间DP)
题目:http://www.lydsy.com:808/JudgeOnline/problem.php?id=1564 分析: 首先因为每个点的数据值不变,所以无论树的形态如何变,树的中序遍历肯定不变 ...
- 1563: [NOI2009]诗人小G
1563: [NOI2009]诗人小G https://lydsy.com/JudgeOnline/problem.php?id=1563 分析: 直接转移f[i]=f[j]+cost(i,j),co ...
- 【BZOJ1564】【NOI2009】二叉查找树(动态规划)
[BZOJ1564][NOI2009]二叉查找树(动态规划) 题面 BZOJ 洛谷 题目描述 已知一棵特殊的二叉查找树.根据定义,该二叉查找树中每个结点的数据值都比它左儿子结点的数据值大,而比它右儿子 ...
- BZOJ_1563_[NOI2009]诗人小G_决策单调性
BZOJ_1563_[NOI2009]诗人小G_决策单调性 Description Input Output 对于每组数据,若最小的不协调度不超过1018,则第一行一个数表示不协调度若最小的不协调度超 ...
随机推荐
- cpu真实核数
判断依据: 1.具有相同core id的cpu是同一个core的超线程. 2.具有相同physical id的cpu是同一颗cpu封装的线程或者cores. 英文版: 1.Physical id an ...
- thinkphp5中Indirect modification of overloaded element of XXX has no effect的解决办法
最近在使用Thinkphp5做foreach循环嵌套的时候报错:Indirect modification of overloaded element of XXX has no effect,网上搜 ...
- 「小程序JAVA实战」小程序的留言和评价功能(70)
转自:https://idig8.com/2018/10/28/xiaochengxujavashizhanxiaochengxudeliuyanhepingjiagongneng69/ 目前小程序这 ...
- 跟我学算法-吴恩达老师(超参数调试, batch归一化, softmax使用,tensorflow框架举例)
1. 在我们学习中,调试超参数是非常重要的. 超参数的调试可以是a学习率,(β1和β2,ε)在Adam梯度下降中使用, layers层数, hidden units 隐藏层的数目, learning_ ...
- Shadow Mapping 的原理与实践 【转】
早在上世纪七十年代末,Williams在他的“Casting Curved Shadows on Curved Surface”一文中提出了名为Shadow Map的阴影生成技术.之后,他人在此基础上 ...
- 关于调试php的socket服务端中遇到的问题及解决办法
今天终于把socket的服务端解决了,期间遇到了很多问题呢~ 1.用cmd运行php的问题: 2.socket_create()函数未定义问题: 3.查看端口的问题. 以下逐一说说解决办法: 1.在c ...
- Django基础学习四_数据库的增删改查
今天主要学习两个东西 1.如何对数据库做增删改查 2.如果将数据库中的数据用html的方式返回到前台 一.对数据库中增删改查操作 1.首先需要先见表,见表的方法我们在“http://www.cnblo ...
- basicHttpBinding
表示一个绑定,Windows Communication Foundation (WCF) 服务可以使用此绑定配置和公开能够与基于 ASMX 的 Web 服务和客户端通信的终结点以及符合 WS-I B ...
- mysql查询赋值、修改拼接字符串
sql中修改字符串类型的字段可以这么拼接:update tbName set UserName='abc'+UserName; 但mysql中就不行了,需要这样:update tbName set U ...
- NLTK和Stanford NLP两个工具的安装配置
这里安装的是两个自然语言处理工具,NLTK和Stanford NLP. 声明:笔者操作系统是Windows10,理论上Windows都可以: 版本号:NLTK 3.2 Stanford NLP 3.6 ...