【刷题】BZOJ 4816 [Sdoi2017]数字表格

Description

Doris刚刚学习了fibonacci数列。用f[i]表示数列的第i项,那么
f[0]=0
f[1]=1
f[n]=f[n-1]+f[n-2],n>=2
Doris用老师的超级计算机生成了一个n×m的表格,第i行第j列的格子中的数是f[gcd(i,j)],其中gcd(i,j)表示i,
j的最大公约数。Doris的表格中共有n×m个数,她想知道这些数的乘积是多少。答案对10^9+7取模。

Input

有多组测试数据。
第一个一个数T,表示数据组数。
接下来T行,每行两个数n,m
T<=1000,1<=n,m<=10^6

Output

输出T行,第i行的数是第i组数据的结果

Sample Input

3
2 3
4 5
6 7

Sample Output

1
6
960

Solution

莫比乌斯反演以及作死地推式子,开始:

[ans=prod_{i=1}^nprod_{j=1}^mf[gcd(i,j)] ]

[displaystyle =prod_{d=1}^nf[d]^{sum_{i=1}^{lfloor frac{n}{d} floor}mu(i)lfloor frac{n}{id} floor lfloor frac{m}{id} floor} ]

套路设(T=id)

[ans=prod_{d=1}^nf[d]^{sum_{d|T}^nmu(frac{T}{d})lfloor frac{n}{T} floorlfloor frac{m}{T} floor} ]

[=prod_{d=1}^nprod_{d|T}^nf[d]^{mu(frac{T}{d})lfloor frac{n}{T} floorlfloor frac{m}{T} floor} ]

[=prod_{T=1}^nprod_{d|T}f[d]^{mu(frac{T}{d})lfloor frac{n}{T} floorlfloor frac{m}{T} floor} ]

[=prod_{T=1}^n(prod_{d|T}f[d]^{mu(frac{T}{d})})^{lfloor frac{n}{T} floorlfloor frac{m}{T} floor} ]

从第二步到第三步,枚举方式之所以能够改变是因为两者的实际意义是一样的。
想一想是为什么。
(prod_{d=1}^nprod_{d|T}^n)含义是枚举数对((d,T)),保证(T)(d)的倍数且(T)的值不超过(n)
(prod_{T=1}^nprod_{d|T})难道不是一样的吗?
回到题目,只要括号里面的东西能求前缀和(前缀积?。。)括号外面的东西显然可以整除分块
那么就考虑括号里面的东西
要考虑吗。。
线性筛筛完后暴力求。。。复杂度(O(nlnn))可以接受
然后?
就做完了

#include<bits/stdc++.h>
#define ll long long
const int MAXN=1000000+10,Mod=1e9+7;
int cnt,prime[MAXN],vis[MAXN],mu[MAXN];
ll g[MAXN],f[MAXN],fn[MAXN],F[MAXN];
template<typename T> inline void read(T &x)
{
	T data=0,w=1;
	char ch=0;
	while(ch!='-'&&(ch<'0'||ch>'9'))ch=getchar();
	if(ch=='-')w=-1,ch=getchar();
	while(ch>='0'&&ch<='9')data=((T)data<<3)+((T)data<<1)+(ch^'0'),ch=getchar();
	x=data*w;
}
template<typename T> inline void write(T x,char c='')
{
	if(x<0)putchar('-'),x=-x;
	if(x>9)write(x/10);
	putchar(x%10+'0');
	if(c!='')putchar(c);
}
template<typename T> inline void chkmin(T &x,T y){x=(y<x?y:x);}
template<typename T> inline void chkmax(T &x,T y){x=(y>x?y:x);}
template<typename T> inline T min(T x,T y){return x<y?x:y;}
template<typename T> inline T max(T x,T y){return x>y?x:y;}
inline ll qexp(ll a,ll b)
{
	ll res=1;
	while(b)
	{
		if(b&1)res=res*a%Mod;
		a=a*a%Mod;
		b>>=1;
	}
	return res;
}
inline void init()
{
	memset(vis,1,sizeof(vis));
	vis[0]=vis[1]=0;
	mu[1]=1;
	g[0]=g[1]=1;
	for(register int i=2;i<MAXN;++i)
	{
		g[i]=1;
		if(vis[i])
		{
			prime[++cnt]=i;
			mu[i]=-1;
		}
		for(register int j=1;j<=cnt&&i*prime[j]<MAXN;++j)
		{
			vis[i*prime[j]]=0;
			if(i%prime[j])mu[i*prime[j]]=-mu[i];
			else break;
		}
	}
	f[1]=fn[1]=1;
	for(register int i=2;i<MAXN;++i)
	{
		f[i]=(f[i-1]+f[i-2])%Mod;
		fn[i]=qexp(f[i],Mod-2);
	}
	for(register int i=1;i<MAXN;++i)
		if(mu[i]!=0)
			for(register int j=i;j<MAXN;j+=i)(g[j]*=(mu[i]==1?f[j/i]:fn[j/i]))%=Mod;
	for(register int i=1;i<MAXN;++i)(g[i]*=g[i-1])%=Mod;
}
inline ll solve(ll n,ll m)
{
	ll res=1;
	if(n>m)std::swap(n,m);
	for(register int i=1;;)
	{
		if(i>n)break;
		int j=min(n/(n/i),m/(m/i));
		(res*=qexp(g[j]*qexp(g[i-1],Mod-2)%Mod,(n/i)*(m/i)%(Mod-1)))%=Mod;
		i=j+1;
	}
	return (res+Mod)%Mod;
}
int main()
{
	init();
	int T;
	read(T);
	while(T--)
	{
		ll n,m;
		read(n);read(m);
		write(solve(n,m),'
');
	}
	return 0;
}
原文地址:https://www.cnblogs.com/hongyj/p/8575668.html