• RMQ(连续相同最大值)


    http://poj.org/problem?id=3368

    Frequent values
    Time Limit: 2000MS   Memory Limit: 65536K
    Total Submissions: 24727   Accepted: 8612

    Description

    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.

    Input

    The input consists of several test cases. Each test case starts with a line containing two integers n and q (1 ≤ n, q ≤ 100000). The next line contains n integers a1 , ... , an (-100000 ≤ ai ≤ 100000, for each i ∈ {1, ..., n}) separated by spaces. You can assume that for each i ∈ {1, ..., n-1}: ai ≤ ai+1. The following q lines contain one query each, consisting of two integers i and j (1 ≤ i ≤ j ≤ n), which indicate the boundary indices for the
    query.

    The last test case is followed by a line containing a single 0.

    Output

    For each query, print one line with one integer: The number of occurrences of the most frequent value within the given range.

    Sample Input

    10 3
    -1 -1 1 1 1 1 3 10 10 10
    2 3
    1 10
    5 10
    0

    Sample Output

    1
    4
    3

    Source

    题意:排好序的序列,询问区间值连续出现最多的次数。st表
    #include <cstdio>
    #include <cstring>
    #include <cmath>
    #include <algorithm>
    #include <iostream>
    #include <algorithm>
    #include <iostream>
    #include<cstdio>
    #include<string>
    #include<cstring>
    #include <stdio.h>
    #include <string.h>
    #include <vector>
    #define ME(x , y) memset(x , y , sizeof(x))
    #define SF(n) scanf("%d" , &n)
    #define rep(i , n) for(int i = 0 ; i < n ; i ++)
    #define INF  0x3f3f3f3f
    using namespace std;
    const int N = 1e5 + 9 ;
    const int M = 1e5 + 9 ;
    int dp[N][40] , a[N] , b[N] ;//b数组统计连续相同个数
    int n , q ;
    
    void RMQ()
    {
        for(int i = 1 ; i <= n ; i++)
        {
            dp[i][0] = b[i];
        }
        for(int j = 1 ; j < 30 ; j++)
        {
            for(int i = 1 ; i <= n ; i++)
            {
                if(i + (1 << j) - 1 <= n)
                {
                    dp[i][j] = max(dp[i][j-1] , dp[i+(1<<(j-1))][j-1]);
                }
            }
        }
    
    }
    
    int query(int l , int r)
    {
        if(r < l) return 0 ;//这个条件不能少,如果查询左右端点相等时,l将大于r
        int ans = 0 ;
        int k = (int)log2(r - l + 1);
        ans = max(dp[l][k] , dp[r - (1 << k) + 1][k]);
        return ans ;
    }
    
    int main()
    {
    
        while(~scanf("%d" , &n) && n)
        {
    
            scanf("%d" , &q);
            //init();
            for(int i = 1 ; i <= n ; i++)
            {
                scanf("%d" , &a[i]);
                b[i] = 1 ;
            }
            for(int i = 2 ; i <= n ; i++)
            {
                if(a[i] == a[i-1])
                {
                    b[i] = b[i-1] + 1 ;
                }
                else
                {
                    b[i] = 1 ;
                }
            }
            RMQ();
            for(int i = 1 ; i <= q ; i++)
            {
                int l ,  r , ans = 0;
                scanf("%d%d" , &l , &r);
                int t = l ;
                while(t <= r && a[t] == a[t-1])//以防左端点被截断
                {
                    t++;
                }
                ans = max(t - l , query(t , r));//左端点被截断的话,需要用值减去下标
                printf("%d
    " , ans);
            }
    
        }
    
    
    
    
    
        return 0 ;
    }

    线段树

    #include <cstdio>
    #include <cstring>
    #include <cmath>
    #include <algorithm>
    #include <iostream>
    #include <algorithm>
    #include <iostream>
    #include<cstdio>
    #include<string>
    #include<cstring>
    #include <stdio.h>
    #include <string.h>
    #include <vector>
    #define ME(x , y) memset(x , y , sizeof(x))
    #define SF(n) scanf("%d" , &n)
    #define rep(i , n) for(int i = 0 ; i < n ; i ++)
    #define INF  0x3f3f3f3f
    using namespace std;
    const int N = 1e5 + 9 ;
    const int M = 1e5 + 9 ;
    int dp[N][40] , a[N] , b[N] ;//b数组统计连续相同个数
    int n , q , k = 1 , ans = 0;
    
    struct node{
        int l , r , val ;
    }tree[N << 2];
    
    void build(int l , int r , int root)
    {
        tree[root].l = l , tree[root].r = r ;
        if(l == r)
        {
            tree[root].val = b[k++];
          //  cout << root << " " << tree[root].val << endl ;
            return ;
        }
        int mid = (l + r) >> 1;
        build(l , mid , root*2);
        build(mid+1 , r , root*2+1);
        tree[root].val = max(tree[root*2].val , tree[root*2+1].val);
    }
    
    void query(int l , int r , int root)
    {
        if(r < l)
            return  ;
        if(tree[root].l >= l && tree[root].r <= r)
        {
            ans = max(tree[root].val , ans) ;
            return ;
        }
        int mid =(tree[root].l + tree[root].r) >> 1;
        if(l <= mid)
            query(l , r , root*2);
        if(r > mid)
            query(l , r , root*2+1);
    }
    
    
    int main()
    {
    
        while(~scanf("%d" , &n) && n)
        {
    
            scanf("%d" , &q);
            //init();
            for(int i = 1 ; i <= n ; i++)
            {
                scanf("%d" , &a[i]);
                b[i] = 1 ;
            }
            for(int i = 2 ; i <= n ; i++)
            {
                if(a[i] == a[i-1])
                {
                    b[i] = b[i-1] + 1 ;
                }
                else
                {
                    b[i] = 1 ;
                }
            }
          //  memset(tree , 0 ,sizeof(tree));
            k = 1 ;
            build(1 , n , 1);
            for(int i = 1 ; i <= q ; i++)
            {
                int l ,  r , sum = 0;
                scanf("%d%d" , &l , &r);
                int t = l ;
                while(t <= r && a[t] == a[t-1])//以防左端点被截断
                {
                    t++;
                }
                query(t , r , 1);
    
                sum = max(ans , t - l);
                printf("%d
    " , sum);
                ans = 0 ;
            }
    
        }
    
    
    
    
    
        return 0 ;
    }
  • 相关阅读:
    OpenLayers 添加OpenStreetMap(OSM)瓦片层示例
    [转]Net Framework引路蜂地图开发示例
    SVG关注复杂图形的网页绘制技术
    解决Easyui1.3.3 IE8兼容性问题
    Ajax学习教程在线阅读
    判断访问站点的浏览器类型
    修改TOMCAT服务器图标为应用LOGO
    ArcGIS学习推荐基础教程摘录
    Jquery html页面处理基础
    android AppWidget的使用以及利用TimerTask实现widget的定时更新
  • 原文地址:https://www.cnblogs.com/nonames/p/11332598.html
Copyright © 2020-2023  润新知