【bzoj4491】我也不知道题目名字是什么 离线扫描线+线段树
题目描述
给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串
输入
第一行n,表示A数组有多少元素
接下来一行为n个整数A[i]
接下来一个整数Q,表示询问数量
接下来Q行,每行2个整数l,r
输出
对于每个询问,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串
样例输入
9
1 2 3 4 5 6 5 4 3
5
1 6
1 7
2 7
1 9
5 9
样例输出
6
6
5
6
4
题解
离线扫描线+线段树
考虑询问 $[l,r]$ ,对于选出子串的左端点i,右端点一定是 $min(pos[i],r)$ ,其中 $pos[i]$ 表示从 $i$ 开始的最长不上升子串的最终位置。
那么我们先扫一遍整个序列,求出从每个位置开始的最长不上升子串的最终位置。
然后把 $min(pos[i],r)$ 分为两种情况,即求满足 $pos[i]\le r$ 的所有i中最大的 $pos[i]-i$ ,和满足 $pos[i]>r$ 的所有 $i$ 中最大的 $r-i$(即最大的 $-i$ )
把序列位置看作 $(i,pos[i])$,询问看作 $(l,r)$ ,相当于二维平面上的点,显然这个问题可以用离线扫描线+线段树的方法来解决。
把序列位置按照 $pos[i]$ 从小到大排序,把询问按照 $r$ 从小到大排序。对于 $pos[i]\le r$ 的情况,从前往后扫描询问,把所有 $pos[i]\le r$ 的 $i$ 在线段树的 $i$ 的位置加入 $pos[i]-i$ ,询问直接在线段树求 $[l,r]$ 的最大值。 $pos[i]>r$的情况同理。
时间复杂度 $O(n\log n)$
#include <cstdio>
#include <algorithm>
#define N 50010
#define lson l , mid , x << 1
#define rson mid + 1 , r , x << 1 | 1
using namespace std;
struct data
{
int l , r , id;
bool operator<(const data &a)const {return r < a.r;}
}v[N << 1] , q[N];
int a[N] , tot , mx[N << 2] , ans[N];
void build(int l , int r , int x)
{
mx[x] = -1 << 30;
if(l == r) return;
int mid = (l + r) >> 1;
build(lson) , build(rson);
}
void update(int p , int v , int l , int r , int x)
{
mx[x] = max(mx[x] , v);
if(l == r) return;
int mid = (l + r) >> 1;
if(p <= mid) update(p , v , lson);
else update(p , v , rson);
}
int query(int b , int e , int l , int r , int x)
{
if(b <= l && r <= e) return mx[x];
int mid = (l + r) >> 1 , ans = -1 << 30;
if(b <= mid) ans = max(ans , query(b , e , lson));
if(e > mid) ans = max(ans , query(b , e , rson));
return ans;
}
int main()
{
int n , m , i , p;
scanf("%d" , &n);
for(i = 1 ; i <= n ; i ++ ) scanf("%d" , &a[i]);
for(p = i = 1 ; i <= n ; i ++ )
if(a[i] < a[i - 1])
while(p < i)
v[++tot].l = p ++ , v[tot].r = i - 1;
while(p <= n) v[++tot].l = p ++ , v[tot].r = n;
for(p = i = 1 ; i <= n ; i ++ )
if(a[i] > a[i - 1])
while(p < i)
v[++tot].l = p ++ , v[tot].r = i - 1;
while(p <= n) v[++tot].l = p ++ , v[tot].r = n;
scanf("%d" , &m);
for(i = 1 ; i <= m ; i ++ ) scanf("%d%d" , &q[i].l , &q[i].r) , q[i].id = i;
sort(v + 1 , v + tot + 1) , sort(q + 1 , q + m + 1);
build(1 , n , 1);
for(p = i = 1 ; i <= m ; i ++ )
{
while(p <= tot && v[p].r <= q[i].r) update(v[p].l , v[p].r - v[p].l + 1 , 1 , n , 1) , p ++ ;
ans[q[i].id] = max(ans[q[i].id] , query(q[i].l , q[i].r , 1 , n , 1));
}
build(1 , n , 1);
for(p = tot , i = m ; i ; i -- )
{
while(p && v[p].r > q[i].r) update(v[p].l , -v[p].l , 1 , n , 1) , p -- ;
ans[q[i].id] = max(ans[q[i].id] , query(q[i].l , q[i].r , 1 , n , 1) + q[i].r + 1);
}
for(i = 1 ; i <= m ; i ++ ) printf("%d\n" , ans[i]);
return 0;
}
【bzoj4491】我也不知道题目名字是什么 离线扫描线+线段树的更多相关文章
- BZOJ4491: 我也不知道题目名字是什么
Description 给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串 Input 第一行n,表示A数组有多少元素接下来一行为n个整数A[i]接下来一 ...
- 2019.03.09 bzoj4491: 我也不知道题目名字是什么(线段树)
传送门 题意:给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串. 思路: 注意要求的是子串而不是子序列!!! 然后直接用线段树维护最大子段和的方式合并一 ...
- 【BZOJ4491】我也不知道题目名字是什么 [线段树]
我也不知道题目名字是什么 Time Limit: 10 Sec Memory Limit: 512 MB[Submit][Status][Discuss] Description 给定一个序列A[i ...
- 【BZOJ4991】我也不知道题目名字是什么(线段树)
[BZOJ4991]我也不知道题目名字是什么(线段树) 题面 BZOJ 题解 对于线段树维护的区间维护以下东西: 区间左(右)端开始(结束)的最长(短)子串的长度 左端右端的值,以及当前区间内的答案 ...
- BZOJ 4491: 我也不知道题目名字是什么
4491: 我也不知道题目名字是什么 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 278 Solved: 154[Submit][Status][ ...
- BZOJ 4491: 我也不知道题目名字是什么 RMQ
4491: 我也不知道题目名字是什么 Time Limit: 10 Sec Memory Limit: 512 MBSubmit: 317 Solved: 174[Submit][Status][ ...
- 【洛谷4770/UOJ395】[NOI2018]你的名字(后缀数组_线段树合并)
题目: 洛谷4770 UOJ395 分析: 一个很好的SAM应用题-- 一句话题意:给定一个字符串\(S\).每次询问给定字符串\(T\)和两个整数\(l\).\(r\),求\(T\)有多少个本质不同 ...
- BZOJ 4491: 我也不知道题目名字是什么 线段树+离线
code: #include <string> #include <cstring> #include <cstdio> #include <algorith ...
- [NOI2018]你的名字(后缀自动机+线段树)
题目描述 小A 被选为了ION2018 的出题人,他精心准备了一道质量十分高的题目,且已经把除了题目命名以外的工作都做好了. 由于ION 已经举办了很多届,所以在题目命名上也是有规定的,ION 命题手 ...
随机推荐
- ProtoBuffer由.proto文件生成.cc/.h
ProtoBuffer由.proto文件生成.cc/.h 一:编译源码下载地址:http://code.google.com/p/protobuf/downloads/list 下载后,根据编译说明进 ...
- ASP.NET MVC 微信JS-SDK认证
layout: post title: ASP.NET MVC 微信JS-SDK认证 category: .net date: 2016-11-01 00:00:00 tags: .net javas ...
- HashMap在并发场景下踩过的坑
本文来自网易云社区 作者:张伟 关于HashMap在并发场景下的问题有很多人,很多公司遇到过!也很多人总结过,我们很多时候都认为这样都坑距离自己很远,自己一定不会掉入这样都坑.可是我们随时都有就遇到了 ...
- 如何用istio实现应用的灰度发布
Istio为用户提供基于微服务的流量治理能力.Istio允许用户按照标准制定一套流量分发规则,并且无侵入的下发到实例中,平滑稳定的实现灰度发布功能. 基于华为云的Istio服务网格技术,使得灰度发布全 ...
- 「日常训练」Equation(HDU-5937)
题意与分析 时隔一个月之后来补题.说写掉的肯定会写掉. 题意是这样的:给1~9这些数字,每个数字有\(X_i\)个,问总共能凑成多少个不同的等式\(A+B=C\)(\(A,B,C\)均为1位,\(1+ ...
- Fiddler使用总结(一)
Fiddler基础知识 .Fiddler是强大的抓包工具,它的原理是以web代理服务器的形式进行工作的,使用的代理地址是:127.0.0.1,端口默认为8888,我们也可以通过设置进行修改. .代理就 ...
- Siki_Unity_0_Unity A计划直播视频
Unity A计划直播视频 2017-07-04直播 任务1:如何识别以招聘来招培训生的公司: 打着招聘的旗号帮培训机构找培训生 关键词:实训生 任务2:如何识别一个公司的好坏和规模大小: 猎聘(中高 ...
- Idea Live Templates
常用live templates 模板 注释 : * * @param $params$ * @return $return$ * $date$ $time$ chiyuanzhen743 */ lo ...
- 通过流的方式操作hadoop的API
通过流的方式操作hadoop的API 功能: 可以直接用来操作hadoop的文件系统 可以用在mapreduce的outputformat中设置RecordWrite 参考: 概念理解 http:// ...
- Java学习笔记-11.运行期间类型鉴定
1.Class对象的getClasses()方法获取的是该类中所有的公共的内部类,以及从父类,父接口继承来的内部类.getinterfaces()方法返回类继承的所有接口. import javax. ...