码迷,mamicode.com
首页 > 其他好文 > 详细

HDU5780 gcd 欧拉函数

时间:2016-08-01 12:11:24      阅读:155      评论:0      收藏:0      [点我收藏+]

标签:

http://acm.hdu.edu.cn/showproblem.php?pid=5780

BC #85 1005

思路:

首先原式化简:x?^gcd(a,b)??1

也就是求n内,(公约数是i的对数)*x^i-1的和,其中i为n内的两两最大公约数。那么问题可以转化成先预处理出i,再求和,注意O(n*300)=1,正常情况会卡常数。必须还要优化

由于 ans=∑s[d](x^?d??1),记s[d]=最大公约数为d的对数

我们注意到求s[d] or (公约数是i的对数),也就是求n/i以内互质数的对数,显然用欧拉来做

s[d]=2*(phi[1]+phi[2]+...+phi[n/d])-1

注意到:d不同,但是n/d一样,也就是s[d]可能有多个相同,比如 10/6 10/7 10/8 10/9 10/10,所以求s[d]相同的项,我们可以用等比公式求和(快速幂+逆元 新知识)

所以我们只要找到每一段s[d]就可以 即 j=n/(n/i),j为最后一个相同s[d]的下标

 1 // #pragma comment(linker, "/STACK:102c000000,102c000000")
 2 #include <iostream>
 3 #include <cstdio>
 4 #include <cstring>
 5 #include <sstream>
 6 #include <string>
 7 #include <algorithm>
 8 #include <list>
 9 #include <map>
10 #include <vector>
11 #include <queue>
12 #include <stack>
13 #include <cmath>
14 #include <cstdlib>
15 // #include <conio.h>
16 using namespace std;
17 #define pi acos(-1.0)
18 const int N = 1e6+10;
19 const int MOD = 1e9+7;
20 #define inf 0x7fffffff
21 typedef long long  LL;
22 
23 void fre() { freopen("in.txt","r",stdin);}
24 inline int read(){int x=0,f=1;char ch=getchar();while(ch>9||ch<0) {if(ch==-) f=-1;ch=getchar();}while(ch>=0&&ch<=9) { x=x*10+ch-0;ch=getchar();}return x*f;}
25 
26 LL pow_m(LL x,LL n) 
27 {
28     LL res=1;
29     while(n>0)
30     {
31         if(n & 1)
32             res=(res*x)%MOD;
33         x=(x*x)%MOD;
34         n >>= 1;
35     }
36     return res;
37 }
38 
39 int prime[N],sphi[N];
40 int inv[N];
41 void e_fun(){
42     sphi[1]=1;
43     for(int i=2;i<=N;i++){
44         if(!sphi[i]){
45             prime[++prime[0]]=i;
46             sphi[i]=i-1;
47         }
48         for(int j=1;j<=prime[0]&&i*prime[j]<=N;j++){
49             if(i%prime[j]) sphi[i*prime[j]]=sphi[i]*(prime[j]-1);
50             else sphi[i*prime[j]]=sphi[i]*prime[j];
51         }
52     }
53     for(int i=1;i<=N;i++) sphi[i]=(sphi[i-1]+sphi[i])%MOD;
54     
55     //打表求逆元 
56     // inv[1] = inv[0] = 1;
57     // for(int i = 2;i < N;i++)
58     // inv[i] = inv[MOD%i]*(MOD-MOD/i)%MOD;
59 }
60 
61 
62 void ex_gcd(LL a,LL b,LL &d,LL &x,LL &y) {
63     if (!b) {
64         d = a;
65         x = 1;
66         y = 0;
67     }
68     else {
69         ex_gcd(b, a%b, d, y, x);
70         y -= x*(a/b);
71     }
72     // return x;
73 }
74 
75 LL sn(LL q,LL n){
76     if(q==1) return n;
77     LL x,y,d;
78     ex_gcd(q-1,MOD,d,x,y);
79     return (pow_m(q,n)-1)*((x+MOD)%MOD)%MOD;
80 }
81 
82 int main(){
83     e_fun();
84     int T;
85     T=read();
86     while(T--){
87         int x,n;
88         x=read(),n=read();
89         LL ans=0;
90         for(int i=1,j;i<=n;i=j+1){
91             j=n/(n/i);
92             LL sd=2*sphi[n/i]-1;
93             ans=(ans + sd*(pow_m(x,i)*sn(x,j-i+1)%MOD -(j-i+1))%MOD) % MOD;
94         }
95         printf("%I64d\n",ans);
96     }
97     return 0;
98 }

 

HDU5780 gcd 欧拉函数

标签:

原文地址:http://www.cnblogs.com/ITUPC/p/5725041.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!