评论

收藏

[C++] 【BZOJ】【2190】【SDOI2008】仪仗队

编程语言 编程语言 发布于:2021-08-04 15:50 | 阅读数:227 | 评论:0

欧拉函数/莫比乌斯函数Orz iwtwiioi
这个嘛……很明显在同一条线上的两个点一定是满足  x1*k=x2,y1*k=y2,(好吧这个表示方式有点傻逼,懂得就好了)那么这条线上的点只有第一个会被看到,即x,y互质的那一个点(如果gcd(x,y)==k>1那么肯定在它前面还有点(x/k,y/k) )
但是马上你就会指着鼻子骂我说这特么不是胡扯么……(2,1) 和 (4,2)明明都能被看到!那是因为这里的“坐标原点”是(1,1)啊……所以坐标都减个1就好了→_→
所以就是求 ………………看iwtwiioi的博客去吧- -b 我不会用LeTeX……
嗯就是求:(原谅我截图跑……iwtwiioi实在太神了)
DSC0000.jpg

这样一个东西……就是对于每个x,求一共有多少个y与它互质,这不就是欧拉函数的定义么→_→
所以可以写成
DSC0001.jpg 【然后再乘二(x和y交换算两组)】

最后还有一点细节要注意:(1,1)这个点统计了两次,(1,0) 和 (0,1)这两个点没有算。
还有一种姿势,如果是用莫比乌斯函数写的话,第一个和式还可以写成:【Orz lqybzx】
sigma{ mu*(n/i)*(n/i) }  (不要吐槽我的语法……)
我的代码:(线性筛模板参见《线性筛法与积性函数》——贾志鹏)
1 /**************************************************************
 2   Problem: 2190
 3   User: Tunix
 4   Language: C++
 5   Result: Accepted
 6   Time:24 ms
 7   Memory:1624 kb
 8 ****************************************************************/
 9  
10 //BZOJ 2190
11 #include<cstdio>
12 #include<cstring>
13 #include<cstdlib>
14 #include<iostream>
15 #include<algorithm>
16 #define rep(i,n) for(int i=0;i<n;++i)
17 #define F(i,j,n) for(int i=j;i<=n;++i)
18 #define D(i,j,n) for(int i=j;i>=n;--i)
19 using namespace std;
20 int getint(){
21   int v=0,sign=1; char ch=getchar();
22   while(ch<'0'||ch>'9'){ if (ch=='-') sign=-1; ch=getchar();}
23   while(ch>='0'&&ch<='9'){ v=v*10+ch-'0'; ch=getchar();}
24   return v*=sign;
25 }
26 /******************tamplate*********************/
27 const int N=40010;
28 int phi[N],prime[N];
29 bool check[N];
30 void getphi(int n){
31   memset(check,0,sizeof check);
32   phi[1]=1;
33   int tot=0;
34   F(i,2,n){
35     if(!check[i]){
36       prime[tot++]=i;
37       phi[i]=i-1;
38     }
39     rep(j,n){
40       if(i*prime[j]>n) break;
41       check[i*prime[j]]=1;
42       if(i % prime[j]==0){
43         phi[i*prime[j]]=phi[i]*prime[j];
44         break;
45       }
46       else phi[i*prime[j]]=phi[i]*(prime[j]-1);
47     }
48   }
49 }
50 int main(){
51   int n=getint(),ans=0;
52   getphi(n);
53   F(i,1,n-1)
54     ans+=phi[i];
55   ans=ans*2+1;
56   printf("%d\n",ans);
57   return 0;
58 }
欧拉函数
1 /**************************************************************
 2   Problem: 2190
 3   User: Tunix
 4   Language: C++
 5   Result: Accepted
 6   Time:28 ms
 7   Memory:1624 kb
 8 ****************************************************************/
 9  
10 //BZOJ 2190
11 #include<cstdio>
12 #include<cstring>
13 #include<cstdlib>
14 #include<iostream>
15 #include<algorithm>
16 #define rep(i,n) for(int i=0;i<n;++i)
17 #define F(i,j,n) for(int i=j;i<=n;++i)
18 #define D(i,j,n) for(int i=j;i>=n;--i)
19 using namespace std;
20 int getint(){
21   int v=0,sign=1; char ch=getchar();
22   while(!isdigit(ch)) {if(ch=='-') sign=-1; ch=getchar();}
23   while(isdigit(ch))  {v=v*10+ch-'0'; ch=getchar();}
24   return v*sign;
25 }
26 /*******************template********************/
27 const int N=40001;
28 int prime[N],mu[N];
29 bool check[N];
30 void getmu(int n){
31   memset(check,0,sizeof check);
32   mu[1]=1;
33   int tot=0;
34   F(i,2,n){
35     if(!check[i]){
36       prime[tot++]=i;
37       mu[i]=-1;
38     }
39     rep(j,tot){
40       if(i*prime[j]>n) break;
41       check[i*prime[j]]=true;
42       if(i%prime[j]==0){
43         mu[i*prime[j]]=0;
44         break;
45       }
46       else mu[i*prime[j]]=-mu[i];
47     }
48   }
49 }
50 int main(){
51   int n,ans=0;
52   n=getint();
53   n--;
54   getmu(n);
55   F(i,1,n) ans+=mu[i]*(n/i)*(n/i);
56   ans=ans+2;
57   printf("%d\n",ans);
58   return 0;
59 }
60
莫比乌斯函数

关注下面的标签,发现更多相似文章