• BZOJ 4028 分块


    zrt当年是怎么想到的…….
    思路:
    考虑把序列分块
    对于每块 存xor[i] 表示从本块开头到i的前缀异或和
    把它扔进set里
    存gcd[i]表示从本块开头到i的前缀gcd.
    如果这一块的GCD和整个的gcd的gcd是一样的 从set里找ans
    否则暴力..
    GCD最多log种 所以是复杂度是O(nsqrt(n)logn)的

    //By SiriusRen
    #include <cmath>
    #include <set>
    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    using namespace std;
    const int N=100050;
    int n,q,a[N],Block,block[N],XOR[N],GCD[N];
    char op[150];
    typedef long long ll;
    set<int>s[320];
    int gcd(int x,int y){return y?gcd(y,x%y):x;}
    signed main(){
        scanf("%d",&n),Block=sqrt(n);
        for(int i=0;i<n;i++)scanf("%d",&a[i]),block[i]=i/Block+1;
        for(int i=0;i<n;i++){
            if(block[i]==block[i-1])XOR[i]=XOR[i-1]^a[i],GCD[i]=gcd(a[i],GCD[i-1]);
            else XOR[i]=a[i],GCD[i]=a[i];
            s[block[i]].insert(XOR[i]);
        }
        scanf("%d",&q);
        while(q--){
            scanf("%s",op);
            if(op[0]=='M'){
                int xx,yy;
                scanf("%d%d",&xx,&yy),a[xx]=yy;
                s[block[xx]].clear();
                int tp=lower_bound(block,block+n,block[xx])-block;
                XOR[tp]=a[tp],GCD[tp]=a[tp];s[block[xx]].insert(XOR[tp]);
                for(int i=tp+1;block[i]==block[tp];i++)
                    XOR[i]=XOR[i-1]^a[i],GCD[i]=gcd(a[i],GCD[i-1]),s[block[xx]].insert(XOR[i]);
            }
            else{
                ll Q;int Gcd=GCD[lower_bound(block,block+n,2)-block-1],Xor=XOR[lower_bound(block,block+n,2)-block-1];
                scanf("%lld",&Q);
                for(int i=0;block[i]==1;i++)
                    if((ll)GCD[i]*XOR[i]==Q){printf("%d
    ",i);goto ed;}
                for(int i=2;i<=block[n-1];i++){
                    if(gcd(GCD[upper_bound(block,block+n,i)-block-1],Gcd)!=Gcd){
                        int tp=lower_bound(block,block+n,i)-block;
                        for(int j=tp;block[j]==block[tp];j++){
                            Gcd=gcd(Gcd,a[j]),Xor^=a[j];
                            if((ll)Gcd*Xor==Q){printf("%d
    ",j);goto ed;}
                        }
                        continue;
                    }
                    ll temp=(Q/Gcd)^Xor;
                    if(s[i].find(temp)!=s[i].end()){
                        int tp=lower_bound(block,block+n,i)-block;
                        for(int j=tp;block[j]==block[tp];j++)
                            if(XOR[j]==temp){printf("%d
    ",j);goto ed;}
                    }
                    Xor^=XOR[upper_bound(block,block+n,i)-block-1];
                }
                puts("no");
            }ed:;
        }
    }
  • 相关阅读:
    [Java] [Exception]
    [Go back to REDIS]
    [Java] [内存泄露]
    [ZK] [Related Materials]
    [Scala] [Coursera]
    <zk在大型分布式系统中的应用>
    [Java] [Lock] [Synchronized VS ReentrantLock]
    [Data Structure] Tree
    投影矩阵的计算过程
    SQL Server 2012
  • 原文地址:https://www.cnblogs.com/SiriusRen/p/6532023.html
Copyright © 2020-2023  润新知