POJ 2888 Magic Bracelet [Polya 矩阵乘法]

时间:2023-03-09 19:27:33
POJ 2888 Magic Bracelet [Polya 矩阵乘法]

传送门

题意:竟然扯到哈利波特了....

和上一题差不多,但颜色数很少,给出不能相邻的颜色对


可以相邻的连边建图矩阵乘法求回路个数就得到$f(i)$了....

感觉这样的环上有限制问题挺套路的...旋转的等价循环个数$t$我们很清楚了,并且环上每$t$个元素各属于不同的循环,我们只要求出$t$个元素满足限制的方案数就能得到$C(f)$了

然后再加上$gcd$取值讨论就降到根号了

#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<cmath>
using namespace std;
const int N=1e5+,P=;
typedef long long ll;
inline int read(){
char c=getchar();int x=,f=;
while(c<''||c>''){if(c=='-')f=-; c=getchar();}
while(c>=''&&c<=''){x=x*+c-''; c=getchar();}
return x*f;
}
int n,m,k,u,v;
int p[N];
bool notp[N];
void sieve(int n){
for(int i=;i<=n;i++){
if(!notp[i]) p[++p[]]=i;
for(int j=;j<=p[]&&i*p[j]<=n;j++){
notp[i*p[j]]=;
if(i%p[j]==) break;
}
}
}
inline int phi(int n){
int re=n,m=sqrt(n);
for(int i=;i<=p[]&&p[i]<=m&&p[i]<=n;i++) if(n%p[i]==){
re=re/p[i]*(p[i]-);
while(n%p[i]==) n/=p[i];
}
if(n>) re=re/n*(n-);
return re%P;
}
struct Matrix{
int a[][];
int* operator [](int x){return a[x];}
Matrix(){memset(a,,sizeof(a));}
void ini(){for(int i=;i<=;i++) a[i][i]=;}
}a;
Matrix operator *(Matrix a,Matrix b){
Matrix c;
for(int k=;k<=m;k++)
for(int i=;i<=m;i++) if(a[i][k])
for(int j=;j<=m;j++) if(b[k][j])
(c[i][j]+=a[i][k]*b[k][j])%=P;
return c;
}
Matrix operator ^(Matrix a,int b){
Matrix re;re.ini();
for(;b;b>>=,a=a*a)
if(b&) re=re*a;
return re;
}
inline void mod(int &x){if(x>=P) x-=P;}
int f(int x){
Matrix b=a^x;
int re=;
for(int i=;i<=m;i++) mod(re+=b[i][i]);
return re;
}
inline int Pow(int a,int b){
int re=;
a%=P;
for(;b;b>>=,a=a*a%P)
if(b&) re=re*a%P;
return re;
}
inline int Inv(int a){return Pow(a,P-);}
void solve(){
int m=sqrt(n),ans=;
for(int i=;i<=m;i++) if(n%i==){
mod(ans+= f(i)*phi(n/i)%P);
if(i*i!=n) mod(ans+= f(n/i)*phi(i)%P);
}
printf("%d\n",ans*Inv(n)%P);
}
int main(){
freopen("in","r",stdin);
sieve();
int T=read();
while(T--){
n=read();m=read();k=read();
for(int i=;i<=m;i++) for(int j=;j<=m;j++) a[i][j]=;
for(int i=;i<=k;i++){
u=read();v=read();
a[u][v]=a[v][u]=;
}
solve();
}
}