Description
给出nn个数,问从这nn个数中选一非空子集使得乘积是完全平方数的方案数
Input
第一行一整数表示数的个数,之后输入nn个整数
Output
输出方案数,结果模109+7109+7
Sample Input
4
1 1 1 1
Sample Output
15
Solution1
统计ii出现的次数,由于aiai范围很小,11~中的素数只有1919个,设其为p1,...,p19p1,...,p19,则可以用一个1919位0101的状态sta[x]sta[x]表示一个数字xx含有某素因子个数的奇偶性,第ii位是说明xx的素因子分解形式中有奇数个,是00说明的素因子分解形式中有偶数个pipi,以dp[i][j]dp[i][j]表示从11~选取若干数乘起来,乘积状态为jj的方案数,根据用奇数个还是偶数个ii有转移
用奇数个,则乘积的状态jj会变成^sta[i]sta[i],故有dp[i][jdp[i][j^sta[i]]+=2numi−1⋅dp[i−1][j]sta[i]]+=2numi−1⋅dp[i−1][j]
用偶数个ii,对乘积的状态无影响,故有
最终dp[70][0]−1dp[70][0]−1即为答案,减一是要去掉取空集的情况,时间复杂度O(70⋅219)O(70⋅219)
Code1
#include<cstdio>
#include<iostream>
#include<cstring>
#include<algorithm>
#include<cmath>
#include<vector>
#include<queue>
#include<map>
#include<set>
#include<ctime>
using namespace std;
typedef long long ll;
typedef pair<int,int>P;
const int INF=0x3f3f3f3f,maxn=71;
#define mod 1000000007
int p[maxn],vis[maxn],res;
int n,num[maxn],sta[maxn],dp[maxn][(1<<19)+5],f[100005];
void init(int n=70)
{
res=0;
for(int i=2;i<=n;i++)
if(!vis[i])
{
p[res++]=i;
for(int j=2*i;j<=n;j+=i)vis[j]=1;
}
for(int i=1;i<=n;i++)
{
int ii=i;
for(int j=0;j<res&&p[j]<=ii;j++)
while(ii%p[j]==0)
ii/=p[j],sta[i]^=(1<<j);
}
f[0]=1;
for(int i=1;i<=1e5;i++)f[i]=2*f[i-1]%mod;
}
void add(int &x,int y)
{
x=x+y>=mod?x+y-mod:x+y;
}
int main()
{
init();
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
int temp;
scanf("%d",&temp);
num[temp]++;
}
dp[0][0]=1;
int N=1<<19;
for(int i=1;i<=70;i++)
{
if(!num[i])
{
for(int j=0;j<N;j++)
add(dp[i][j],dp[i-1][j]);
}
else
{
for(int j=0;j<N;j++)
{
add(dp[i][j^sta[i]],(ll)f[num[i]-1]*dp[i-1][j]%mod);
add(dp[i][j],(ll)f[num[i]-1]*dp[i-1][j]%mod);
}
}
}
printf("%d\n",(dp[70][0]+mod-1)%mod);
return 0;
}
Solution2
同样的求出每个aiai的状态si=sta[ai]si=sta[ai]得到状态序列,问题转化为从s1,s2,...,sns1,s2,...,sn中选取非空子集使得其异或和是00,考虑线性基,设线性基的维数是,那么从非基的n−mn−m个数中取任意和做异或得到的数字,均可以通过线性基表出,故答案为2n−m−12n−m−1,减一同样是去掉取空集的方案
Code2
#include<cstdio>
using namespace std;
const int maxn=100005;
#define mod 1000000007
int res=0,p[77],vis[77];
int n,a[maxn],f[maxn],sta[77],base[22];
void init(int n=70)
{
for(int i=2;i<=n;i++)
if(!vis[i])
{
p[res++]=i;
for(int j=2*i;j<=n;j+=i)vis[j]=1;
}
for(int i=1;i<=n;i++)
{
int ii=i;
for(int j=0;j<res&&p[j]<=ii;j++)
while(ii%p[j]==0)
ii/=p[j],sta[i]^=(1<<j);
}
f[0]=1;
for(int i=1;i<=1e5;i++)f[i]=2*f[i-1]%mod;
}
int main()
{
init();
scanf("%d",&n);
for(int i=1;i<=n;i++)
{
int temp;
scanf("%d",&temp);
a[i]=sta[temp];
}
for(int i=1;i<=n;i++)
for(int j=21;j>=0;j--)
if(a[i]>>j&1)
{
if(!base[j])
{
base[j]=a[i];
break;
}
else a[i]^=base[j];
}
int m=0;
for(int i=0;i<=21;i++)
if(base[i])m++;
printf("%d\n",f[n-m]-1);
return 0;
}