• 数字对——RMQ+二分答案


    题目描述 
    小H是个善于思考的学生,现在她又在思考一个有关序列的问题。

    她的面前浮现出一个长度为n的序列{ai},她想找出一段区间[L, R](1 <= L <= R <= n)。

    这个特殊区间满足,存在一个k(L <= k <= R),并且对于任意的i(L <= i <= R),ai都能被ak整除。这样的一个特殊区间 [L, R]价值为R - L。

    小H想知道序列中所有特殊区间的最大价值是多少,而有多少个这样的区间呢?这些区间又分别是哪些呢?你能帮助她吧。

    输入 
    第一行,一个整数n.

    第二行,n个整数,代表ai.

    输出 
    第一行两个整数,num和val,表示价值最大的特殊区间的个数以及最大价值。

    第二行num个整数,按升序输出每个价值最大的特殊区间的L.

    样例输入1 

    4 6 9 3 6 
    样例输出1 
    1 3 

    样例输入2 

    2 3 5 7 11 
    样例输出2 
    5 0 
    1 2 3 4 5

    数据范围

    30%: 1 <= n <= 30 , 1 <= ai <= 32.

    60%: 1 <= n <= 3000 , 1 <= ai <= 1024.

    80%: 1 <= n <= 300000 , 1 <= ai <= 1048576.

    100%: 1 <= n <= 500000 , 1 <= ai < 2 ^ 31.

    这道题主要求的是最长区间长度,可以发现它是具有单调性的:对于一段区间,如果它是特殊区间,那么它一定有子区间是特殊区间。因此我们可以二分最长区间长度,对于每个二分出来的答案进行验证。那么怎么验证?可以发现特殊区间有几个很好的性质:1、ak一定是这段区间的最小值,因为如果有一个数ai比ak小,那ai一定不能被ak整除。2、ak一定是区间gcd,因为将所有ai都除以ak后,这段区间就变成了1,b1,b2……这些数gcd显然是1,再乘回ak后gcd就是ak了。那么只要比较一段区间最小值和gcd是否相同就能判断是否是特殊区间了。但如果暴力找区间最小值和区间gcd显然是不行的,因此要用ST表预处理出来g[i][j]和m[i][j]分别表示以j为起点往后2^i个数的gcd和最小值。再对于每个答案O(n)遍历所有起点判断并记录下来每个特殊区间的起点就OK了。每次验证是O(nlogn),总时间复杂度是O(n*(logn)^2)。

    最后附上代码。

    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    #include<iostream>
    #include<cmath>
    using namespace std;
    int g[20][500010];
    int m[20][500010];
    int a[500010];
    int l[500010];
    int n;
    int L,R;
    int ans;
    int cnt;
    int q[500010];
    int flag;
    int gcd(int x,int y)
    {   
        if(x<y)
        {
            swap(x,y);
        }
        if(y!=0)
        {
            return gcd(y,x%y);
        }
        else
        {
            return x;
        }
    }
    bool check(int x)
    {
        cnt=0;
        for(int i=1;i+x-1<=n;i++)
        {
            int s=l[x];
            if(gcd(g[s][i],g[s][i+x-(1<<s)])==min(m[s][i],m[s][i+x-(1<<s)]))
            {
                q[++cnt]=i;
            }
        }
        if(cnt!=0)
        {
            ans=max(ans,x);
            return true;
        }
        return false;
    }
    int main()
    {
        scanf("%d",&n);
        for(int i=1;i<=n;i++)
        {
            scanf("%d",&a[i]);
            g[0][i]=a[i];
            m[0][i]=a[i];
        }
        for(int i=1;(1<<i)<=n;i++)
        {
            for(int j=1;j+(1<<(i-1))<=n;j++)
            {
                g[i][j]=gcd(g[i-1][j],g[i-1][j+(1<<(i-1))]);
                m[i][j]=min(m[i-1][j],m[i-1][j+(1<<(i-1))]);
            }
        }
        for(int i=2;i<=n;i++)
        {
            l[i]=l[i>>1]+1;
        }
        L=1;
        R=n;
        while(L<R-1)
        {
            int mid=(L+R)>>1;
            if(check(mid)==true)
            {
                L=mid;
            }
            else
            {
                R=mid;
            }
        }
        bool o1=check(L);
        bool o2=check(R);
        bool o3=check(ans);
        printf("%d %d
    ",cnt,ans-1);
        for(int i=1;i<=cnt;i++)
        {
            if(flag==0)
            {
                printf("%d",q[i]);
                flag=1;
            }
            else
            {
                printf(" %d",q[i]);
            }
        }
    }
  • 相关阅读:
    svn-Subversion
    英语学习-2020年4月
    自动化-研究
    数据库-存储过程
    未来软件测试的发展趋势
    学习alex---人生导师
    自动化测试-----总结
    接口测试总结
    jmeter-接口测试项目
    接口测试jmeter
  • 原文地址:https://www.cnblogs.com/Khada-Jhin/p/9118631.html
Copyright © 2020-2023  润新知