yzoj2057 x 题解

Posted donkey2603089141

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了yzoj2057 x 题解相关的知识,希望对你有一定的参考价值。

题意:给出一个集合,要求把这个集合分成两部分,使得一个集合中的任一元素都与另一个集合的全部元素都两两互质

暴力

枚举每个元素O(n^2)再暴力判gcd=1,如果非1就放入不同集合内,用并查集维护联通块的个数即可,答案就是联通块个数减2(无空集)

考虑在暴力的基础上优化,我们可以发现一个元素与它的质因数的倍数一定不在同一集合内,我们可以枚举它的质因数,将它的倍数和它划分于用并查集连接,这样就形成了若干联通块,质数可先用线性筛预处理,注意一下1的判断即可。

代码

#include<bits/stdc++.h>
using namespace std;
const int mod=1e9+7,MAX=1e6+10; 
int T;
long long ans,n,a[MAX],v[MAX],p[MAX],fa[MAX],cnt=0,maxn=-1,book[MAX],sum;
void prime()
    for(int i=2;i<=MAX;++i)
        if(!v[i])
            p[++cnt]=i;
            for(int j=i;j<=MAX/i;++j) v[i*j]=1;
          
    

long long find(long long x)
    if(fa[x]==x) return x;
    return fa[x]=find(fa[x]);

long long clu(long long x,long long y)
    long long tmp=1;
    while(y)
        if(y&1) tmp=(tmp*x)%mod;
        x=(x*x)%mod;
        y=y>>1;
    
    return tmp%mod;

int main()
    prime();
    scanf("%d",&T);
    while(T--)
        scanf("%lld",&n);
        memset(a,0,sizeof(a));
        memset(book,0,sizeof(book));
        memset(v,0,sizeof(v));
        ans=0,maxn=-1,sum=0;
        for(int i=1;i<=MAX;++i) fa[i]=i;
        for(int i=1;i<=n;++i)
            scanf("%lld",&a[i]);
            maxn=max(maxn,a[i]);
            if(a[i]==1) sum++;
            v[a[i]]++;
        
        for(int i=1;i<=cnt;++i)
            for(int j=1;j*p[i]<=maxn;++j)
                if(v[j*p[i]])
                    long long fax=find(p[i]);
                    long long fbx=find(j*p[i]);
                    if(fax!=fbx) fa[fax]=fbx;
                
            
        
        for(int i=1;i<=n;++i)
            if(!book[find(a[i])]) ans++,book[find(a[i])]=1;
        
        if(sum>1) ans+=(sum-1);
        ans=clu(2,ans);
        ans=(ans-2+mod)%mod;
        printf("%d\n",ans);
    
    return 0;

以上是关于yzoj2057 x 题解的主要内容,如果未能解决你的问题,请参考以下文章

yzoj 2372 小B的数字 题解

yzoj P2371 爬山 题解

yzoj1657货仓选址 题解

yzoj P2345 战争 题解

yzoj P2349 取数 题解

yzoj P2344 斯卡布罗集市 题解