BZOJ 4491: 我也不知道题目名字是什么

4491: 我也不知道题目名字是什么

Time Limit: 10 Sec  Memory Limit: 512 MB
Submit: 278  Solved: 154
[Submit][Status][Discuss]

Description

给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串

Input

第一行n,表示A数组有多少元素
接下来一行为n个整数A[i]
接下来一个整数Q,表示询问数量
接下来Q行,每行2个整数l,r

Output

对于每个询问,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串

Sample Input

9
1 2 3 4 5 6 5 4 3
5
1 6
1 7
2 7
1 9
5 9

Sample Output

6
6
5
6
4
//样例解释
五个询问分别对应
[1,6][1,6][2,6][1,6][6,9]

HINT

N,Q<=50000

Source

By 一个读错题的沙茶

分析:

其实就是一个很经典的思想...

既然是最长的不下降子串,也就是连续的,那么我们就差分一下,这样就转化成最长的大于等于0的连续子串...线段树维护前后缀和区间最长就好了...

其实写这道题主要目的是记录一下某只智障(我...)的事迹...

交上去怎么都TLE,然后要了数据,发现可以跑出来答案还是对的...但是跑得极其慢.....大概就是100s估计也跑不出来...

然后请来RYC小盆友来查错...我说我再怎么也不能把线段树写成$O(N^2)$的吧...刚说完一秒钟YSQ小盆友指了query的这一行...

inline M query(int l,int r,int tr){
		if(tree[tr].l==r&&tree[tr].r==r)
			return tree[tr];

  

我写成了$tree[tr].l==r$感觉自己要上天....QwQ~~~

代码:

#include<algorithm>
#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;

const int maxn=50000+5;

int n,m,a[maxn],A[maxn];

struct SegmentTree{

	struct M{
		int l,r,len,lmax,rmax,mmax;
	}tree[maxn<<2];

	inline M merge(M a,M b){
		M res;
		res.l=a.l,res.r=b.r,res.len=a.len+b.len;
		res.lmax=a.len==a.lmax?a.len+b.lmax:a.lmax;
		res.rmax=b.len==b.rmax?b.len+a.rmax:b.rmax;
		res.mmax=max(a.rmax+b.lmax,max(a.mmax,b.mmax));
		return res;
	}

	inline void build(int l,int r,int tr){
		tree[tr].l=l;tree[tr].r=r;tree[tr].len=r-l+1;
		if(l==r){
			if(a[l]>=0)
				tree[tr].lmax=tree[tr].rmax=tree[tr].mmax=1;
			else
				tree[tr].lmax=tree[tr].rmax=tree[tr].mmax=0;
			return;
		}
		int mid=(l+r)>>1;
		build(l,mid,tr<<1),build(mid+1,r,tr<<1|1);
		tree[tr]=merge(tree[tr<<1],tree[tr<<1|1]);
	}

	inline M query(int l,int r,int tr){
		if(tree[tr].l==r&&tree[tr].r==r)
			return tree[tr];
		int mid=(tree[tr].l+tree[tr].r)>>1;
		if(r<=mid)
			return query(l,r,tr<<1);
		else if(l>mid)
			return query(l,r,tr<<1|1);
		else
			return merge(query(l,mid,tr<<1),query(mid+1,r,tr<<1|1));
	}

}tr1,tr2;

signed main(void){
	scanf("%d",&n);
	for(int i=1;i<=n;i++) scanf("%d",&A[i]);
	for(int i=1;i<n;i++) a[i]=A[i+1]-A[i];
	tr1.build(1,n-1,1);
	for(int i=1,j=n;i<j;i++,j--) swap(A[i],A[j]);
	for(int i=1;i<n;i++) a[i]=A[i+1]-A[i];
	tr2.build(1,n-1,1);
	scanf("%d",&m);
	for(int i=1,l,r;i<=m;i++){
		scanf("%d%d",&l,&r);
		if(l==r) puts("1");
		else printf("%d\n",max(tr1.query(l,r-1,1).mmax+1,tr2.query(n-r+1,n-l,1).mmax+1));
	}
	return 0;
}

  



By NeighThorn

时间: 2024-10-16 01:17:58

BZOJ 4491: 我也不知道题目名字是什么的相关文章

BZOJ 4491: 我也不知道题目名字是什么 RMQ

4491: 我也不知道题目名字是什么 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 317  Solved: 174[Submit][Status][Discuss] Description 给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串 Input 第一行n,表示A数组有多少元素 接下来一行为n个整数A[i] 接下来一个整数Q,表示询问数量 接下来Q行,每行2个整数l,r Output 对于每个

【BZOJ4491】我也不知道题目名字是什么 [线段树]

4491: 我也不知道题目名字是什么 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 315  Solved: 173[Submit][Status][Discuss] Description 给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串 Input 第一行n,表示A数组有多少元素 接下来一行为n个整数A[i] 接下来一个整数Q,表示询问数量 接下来Q行,每行2个整数l,r Output 对于每个

bzoj4491 我也不知道题目名字是什么

Description 给定一个序列A[i],每次询问l,r,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串 Input 第一行n,表示A数组有多少元素接下来一行为n个整数A[i]接下来一个整数Q,表示询问数量接下来Q行,每行2个整数l,r Output 对于每个询问,求[l,r]内最长子串,使得该子串为不上升子串或不下降子串 Sample Input 9 1 2 3 4 5 6 5 4 3 5 1 6 1 7 2 7 1 9 5 9 Sample Output 6 6 5 6 4

hdu2191 悼念512汶川大地震遇难同胞——珍惜现在,感恩生活 (这个只是题目名字) (多重背包)

本文出自:http://blog.csdn.net/svitter 原题:http://acm.hdu.edu.cn/showproblem.php?pid=2191 题意:多重背包问题.转换成为01背包解.多重背包转化为01背包的关键在于把件数从整体中孤立出来作为一个新的个体,也就是说不管分类,有多少件就有多少种. AC代码: //============================================================================ // Na

BZOJ——1622: [Usaco2008 Open]Word Power 名字的能量

http://www.lydsy.com/JudgeOnline/problem.php?id=1622 Description 约翰想要计算他那N(1≤N≤1000)只奶牛的名字的能量.每只奶牛的名字由不超过1000个字待构成,没有一个名字是空字体串,  约翰有一张“能量字符串表”,上面有M(1≤M≤100)个代表能量的字符串.每个字符串由不超过30个字体构成,同样不存在空字符串.一个奶牛的名字蕴含多少个能量字符串,这个名字就有多少能量.所谓“蕴含”,是指某个能量字符串的所有字符都在名字串中按

BZOJ 4085 丧心病狂的毒瘤题目

思路: 一眼矩阵快速幂 再用线段树维护一下矩阵就完了... 我hhhhh    哎我还是too young,too simple 入了这个大坑 线段树维护9个值 以上 如果A+1   转移矩阵是这个样子的 B+1 A-1 B-1 同理行么.....

bzoj 1622: [Usaco2008 Open]Word Power 名字的能量【模拟】

模拟即可,注意包含可以是不连续的 方便起见读入的时候全转成小写 #include<iostream> #include<cstdio> using namespace std; const int N=1005; int n,m,s1,s2,ans[N]; struct qwe { int len; char s[N]; }c[N],p[N]; int main() { scanf("%d%d",&n,&m); for(int i=1;i<

TF-IDF算法确定阅读主题词解答英语阅读Title题目

对文章best title的选项进行打分 #include <windows.h> #include <math.h> #include <time.h> #include <stdlib.h> #include <iostream> using namespace std; #define N 5269 //文献数目 #define textN 10 //题目数目 #define ERROR 1 #define OK 0 const int W

[BZOJ 3218]a + b Problem

又是一道主席树优化网络流的好题 按约大爷的教导,源点为白,汇点为黑,搞成最小割 发现暴力连边要爆炸,但是要连的点在线段树中都构成了一个区间,果断主席树优化之 为什么不用一般线段树? 因为要满足 j<i ,这里的可持久化并不是为了查询过去的值,而是为了保留过去的值不与后来弄混- 如果有两个点的 a[i] 相同在线段树里怎么搞? 很简单,从 a[i] 向 a[j] 连一条 inf 的流即可 不过——为什么题目名字那么坑啊啊啊啊啊啊?!!!!! 这种题是不是非要来卡空间不然不痛快是吗?!!!!! 1