POJ 3368 Frequent values (ST表)

题面

You are given a sequence of n integers a1 , a2 , ... , an in non-decreasing order. In addition to that, you are given several queries consisting of indices i and j (1 ≤ i ≤ j ≤ n). For each query, determine the most frequent value among the integers ai , ... , aj.

题解

给你一个序列,然后有m次的查询,每次查询给你左右端点,问你区间内的最长连续序列是多长(元素全部相同)。区间询问,考虑st表,那么我们统计每个序列是以它为值的答案序列的第几项。然后因为是有序序列,所以查询的时候先二分出大于他的哪个元素下标,然后对前一个与右边界比较,如果大于那么就询问的区间长度就是答案吗,如果不是的话,那么我们就比较二分到的长度,和后半部分的答案,去取max,后半部分的答案可以用st表来维护(维护num值)。

代码实现

#include<cstdio>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<vector>
#include<cstring>
using namespace std;
typedef long long ll;
const int maxn=1e5+10;
const int inf=0x3f3f3f3f;
typedef pair<int,int> PII;
void check_min (int &a,int b) {a=min (a,b);}
void check_max (int &a,int b) {a=max (a,b);}
int n,m;
int data[maxn],num[maxn],dp[20][maxn];

inline void RMQ () {
  for (int i=1;i<=n;i++) dp[0][i]=num[i];
  int k=__lg (n+1);
  for (int j=1;j<=k;j++) 
   for (int i=1;i+(1<<j-1)<=n;i++) {
    dp[j][i]=max (dp[j-1][i],dp[j-1][i+(1<<(j-1))]);
   }
  return ;
}

int ask (int x,int y) {
  int k=__lg (y-x+1);
  return max (dp[k][x],dp[k][y-(1<<k)+1]);
}

int main () {
  while (scanf ("%d",&n)) {
    memset (num,0,sizeof (num));
    // for (int i=1;i<=n;i++) printf ("%d ",num[i]);
    // printf ("\n");
    if (n==0) break;
    scanf ("%d",&m);
    for (int i=1;i<=n;i++) scanf ("%d",&data[i]);
    for (int i=1;i<=n;i++) {
      if (i==1||data[i]!=data[i-1]) num[i]=1;
      else num[i]=num[i-1]+1;
    }
    RMQ ();
    // for (int i=1;i<=n;i++) printf ("%d ",&num[i]);
    // printf ("\n");
    while (m--) {
      int a,b,ans=1;
      scanf ("%d%d",&a,&b);
      int pos=upper_bound (data+1,data+1+n,data[a])-data;
      if (pos>b) ans=b-a+1;
      else {
        ans=pos-a;
        check_max (ans,ask (pos,b));
      }
      printf ("%d\n",ans);
    }
  }
  return 0;
}
posted @ 2020-11-17 18:47  Luglucky  阅读(129)  评论(0编辑  收藏  举报