首页 > 其他 > 详细

UVA 11235 RMQ算法

时间:2014-06-16 06:19:40      阅读:407      评论:0      收藏:0      [点我收藏+]

上次的湘潭赛的C题,用线段树敲了下还是WA,不知道为何,我已经注意了处理相同数据,然后他们当时用的RMQ。

所以学了下RMQ,感觉算法思想是一样的,RMQ用了DP或者是递推,由单个数到2^k往上推,虽然有部分重叠的,也没关系,因为RMQ是求区间最值嘛

然后这道题目,要把出现次数化为最值,构造一个新的数组来存贮出现次数,每次是在这个数组里面查询。细节处理要注意,比如所求的区间可能就在一个段里面,需要单独判断。

bubuko.com,布布扣
#include <iostream>
#include <cstdio>
#include <vector>
#define N 100010
using namespace std;
int n,q;
int A[N],num[N],lefts[N],rights[N];
int d[N][20];
void RMQ_init(const vector<int>& a)
{
    int s=a.size();
    for (int i=0;i<s;i++) d[i][0]=a[i];
    for (int j=1;(1<<j)<=n;j++){
        for (int i=0;i+(1<<j)-1<n;i++){
            d[i][j]=max(d[i][j-1],d[i+(1<<(j-1))][j-1]);
        }
    }
}
int RMQ(int L,int R){
    int k=0;
    while ((1<<(k+1))<=R-L+1) k++;
    return max(d[L][k],d[R-(1<<k)+1][k]);
}
int main()
{
    while (scanf("%d",&n))
    {
        if (!n) break;
        scanf("%d",&q);
        vector<int> tmp;
        int st=0;
        for (int i=0;i<n;i++)scanf("%d",&A[i]);
        A[n]=A[n-1]+1;
        for (int i=1;i<=n;i++){
            if (A[i]!=A[i-1]){
                tmp.push_back(i-st);
                for (int j=st;j<i;j++){
                    num[j]=tmp.size()-1;
                    lefts[j]=st;
                    rights[j]=i-1;
                }
                st=i;
            }
        }
        RMQ_init(tmp);
        int L,R;
        while (q--)
        {
            scanf("%d%d",&L,&R);
            L--;R--;
            int ans;
            if (num[L]==num[R]) ans=R-L+1;
            else{
             ans=max(R-lefts[R]+1,rights[L]-L+1);
             if (num[L]+1<num[R]){
                ans=max(ans,RMQ(num[L]+1,num[R]-1));
             }
            }
            printf("%d\n",ans);
        }
    }
    return 0;
}
bubuko.com,布布扣

 

UVA 11235 RMQ算法,布布扣,bubuko.com

UVA 11235 RMQ算法

原文:http://www.cnblogs.com/kkrisen/p/3784039.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!