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

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

Time Limit: 10 Sec  Memory Limit: 512 MB
Submit: 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

对于每个询问,求[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 一个读错题的沙茶

想法:每个点存下$L_i$往左边最长合法,$R_i$往右边最长合法。$[l,r]$的答案即为$max\{R[l],R[l+1]...R[r-L[r]],min(L[r],r-l+1)\}$

用RMQ解决区间最值。

如果这道题有修改怎么做?需要支持区间赋值,求区间最值的线段树。

#include<cstdio>

typedef long long ll;
template<class T>
inline void read(T&x)
{
    x=0;bool f=0;char c=getchar();
    while((c<‘0‘||c>‘9‘)&&c!=‘-‘) c=getchar();if(c==‘-‘)f=1, c=getchar();
    while(c>=‘0‘&&c<=‘9‘){x=x*10+c-‘0‘; c=getchar();}
    x=f?-x:x;
}
const int MAXN(50010);
int n,l,r,Ans,a[MAXN],T[MAXN],R[MAXN],L[MAXN],Q;
int max(int a,int b){return a>b?a:b;}
int F[20][MAXN],logg[MAXN];
void DealRMQ()
{
    for(int i=2;i<=n;i++)logg[i]=logg[i>>1]+1;
    for(int i=1;i<=n;i++)F[0][i]=R[i];
    for(int j=1;j<=logg[n];j++)
    for(int i=1;i<=n;i++)
    {
        int w=1<<(j-1); if(i+w>n)break;
        F[j][i]=max(F[j-1][i],F[j-1][i+w]);
    }
}
int Ask(int l,int r)
{
    int k=logg[r-l+1];   int w=1<<k;
//    fprintf(stderr,"%d %d\n",F[k][l],F[k][r-w+1]);
    return max(F[k][l],F[k][r-w+1]);
}
int main()
{
//    freopen("C.in","r",stdin);
    read(n);
    for(int i=1;i<=n;i++)read(a[i]);
    for(int i=n;i>=1;i--) T[i]=1+(a[i+1]>=a[i])*T[i+1],R[i]=max(R[i],T[i]);
    for(int i=n;i>=1;i--) T[i]=1+(a[i+1]<=a[i])*T[i+1],R[i]=max(R[i],T[i]);
    for(int i=1;i<=n;i++) T[i]=1+(a[i-1]<=a[i])*T[i-1],L[i]=max(L[i],T[i]);
    for(int i=1;i<=n;i++) T[i]=1+(a[i-1]>=a[i])*T[i-1],L[i]=max(L[i],T[i]);
//    for(int i=1;i<=n;i++)
//    printf("i:%d\n L:%d\n  R:%d\n",i,L[i],R[i]);
    DealRMQ();
    read(Q);
    for(int i=1;i<=Q;i++)
    {
        read(l);read(r);
        if(r-L[r]+1<=l)Ans=r-l+1;
        else
        {
            Ans=L[r];     r=r-L[r];
//            fprintf(stderr,"l:%d r%d\n",l,r);
            Ans=max(Ans,Ask(l,r));
        }
        printf("%d\n",Ans);
    }
    return 0;
}
时间: 2024-08-11 00:41:52

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

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

4491: 我也不知道题目名字是什么 Time Limit: 10 Sec  Memory Limit: 512 MBSubmit: 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 对于每个询问,

【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<

HDOJ 题目5289 Assignment(RMQ,技巧)

Assignment Time Limit: 4000/2000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others) Total Submission(s): 1870    Accepted Submission(s): 916 Problem Description Tom owns a company and he is the boss. There are n staffs which are numbered f

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