【bzoj 4833】[Lydsy1704月赛]最小公倍佩尔数

Description

令 $(1+\sqrt 2)^n=e(n)+\sqrt 2\cdot f(n)$ ,其中 $e(n),f(n)$ 都是整数,显然有 $(1-\sqrt 2)^n=e(n)-\sqrt 2\cdot f(n)$。令 $g(n)$ 表示 $f(1),f(2),\cdots ,f(n)$ 的最小公倍数,给定两个正整数 $n$ 和 $p$ ,其中 $p$ 是质数,并且保证 $f(1),f(2),\cdots ,f(n)$ 在模 $p$ 意义下均不为 $0$,请计算 $\sum _{i=1}^{n}i\cdot g(i)$ 模 $p$ 的值。

Input

第一行包含一个正整数 $T$ ,表示有 $T$ 组数据,满足 $T\leq 2^{10}$ 。接下来是测试数据。每组测试数据只占一行,包含两个正整数 $n$$p$ ,满足 $1\leq n\leq 10^6,2\leq p\leq 10^9+7$ 。保证所有测试数据的 $n$ 之和不超过 $3\times 10^6$

Output

对于每组测试数据,输出一行一个非负整数,表示这组数据的答案。

 

在开始推导前先观察两个式子:

$$gcd(fib(a),fib(b))=fib(gcd(a,b))$$

$$gcd(x^a-1,x^b-1)=x^{gcd(a,b)}-1$$

形如 $f(n)=a\cdot f(n-1)+b\cdot f(n-2)$ 的式子具有性质 $gcd(f(x),f(y))=f(gcd(x,y))$

而题目中的式子等价于: $f(0)=0,f(1)=1,f(n)=2f(n-1)+f(n-2)$,同样满足这个性质。

(以下集合 $T$ 均满足 $T\neq \varnothing $

再由式子:$$lcm(S)=\prod_{T\subset S}gcd(T)^{(-1)^{|T|+1}}$$

可以得到:$$g(n)=\prod _{T\subset 2^{[n]}}f(gcd_{i\in T}(i))^{(-1)^{|T|+1}}$$

构造出 $h$ 满足 $f(n)=\prod _{d|n}h(d)$

得到式子:$$\begin{align*} g(n)&=\prod _{T\subset 2^{[n]}}\left ( \prod _{d|gcd_{i\in T}(i)}h(d) \right )^{(-1)^{|T|+1}}\\ &=\prod _{d=1}^{n}h(d)^{\sum _{T\subset 2^{~[\lfloor \frac{n}{d}\rfloor ]~}}(-1)^{|T|+1}} \end{align*}$$

又由二项式定理可证:$$\sum _{T\subset 2^{[\lfloor \frac{n}{d}\rfloor ]}}(-1)^{|T|+1}=-\sum _{i=1}^{\frac{n}{d}}(-1)^i\binom{\frac{n}{d}}{i}=1$$

所以 $g(n)=\prod _{d=1}^{n}h(d)$

问题解决,时间复杂度 $O(nlogn)$

 

 1 #include<cstdio>
 2 #include<algorithm>
 3 #include<cstring>
 4 #define LL long long
 5 using namespace std;
 6 const int N=1e6+5;
 7 int T,n,mod,inv,sum,ans,f[N],h[N];
 8 int read()
 9 {
10     int x=0,f=1;char c=getchar();
11     while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
12     while(c>='0'&&c<='9'){x=x*10+c-'0';c=getchar();}
13     return x*f;
14 }
15 int power(int a,int b)
16 {
17     int ans=1;
18     while(b)
19     {
20         if(b&1)ans=1ll*ans*a%mod;
21         a=1ll*a*a%mod;b>>=1;
22     }
23     return ans;
24 }
25 int main()
26 {
27     T=read();
28     while(T--)
29     {
30         n=read();mod=read();
31         f[0]=0;h[1]=f[1]=1;
32         for(int i=2;i<=n;i++)
33             h[i]=f[i]=(1ll*f[i-1]*2+f[i-2])%mod;
34         for(int i=1;i<=n;i++)
35         {
36             inv=power(h[i],mod-2);
37             for(int j=i+i;j<=n;j+=i)h[j]=1ll*h[j]*inv%mod;
38         }
39         sum=1;ans=0;
40         for(int i=1;i<=n;i++)
41             sum=1ll*sum*h[i]%mod,ans=(ans+1ll*sum*i)%mod;
42         printf("%d\n",ans);
43     }
44     return 0;
45 }
View Code

 

posted @ 2018-04-23 19:05  Zsnuo  阅读(463)  评论(0编辑  收藏  举报