2019.10.28 CSP%您赛第四场t3

  • 2019 年 10 月 28 日
  • 筆記

我写不动前两个了。
原谅一下。
____________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________________

【题目描述】
(varphi)函数是数论中非常常用的函数。对于正整数 x,(varphi)(x) 表示不超过 x 的所有正整数
与 x 互质的个数。
现在我们对它进行一次拓展:对于正整数 x,y,定义 (varphi)(x,y) 表示不超过 y 的所有正
整数与 x 互质的个数。
现在我们给定正整数 n 和 m,对于所有不超过 n 的正整数 i,求 (varphi)(i,m)。
【输入格式】
从文件 phi.in 中读入数据。
输入仅一行两个正整数 n 和 m。
【输出格式】
输出到文件 phi.out 中。
输出 n 行,每行一个整数。第 i 行表示 (varphi)(i,m)。
【样例输入】
11 10
【样例输出】
10
5
7
5
8
3
9
5
7
4
10

以上题面。
顺便补一句,以上算法可接受的复杂度是(O(nsqrt n))
容易知道,我们枚举所有(ileq n)时,时间复杂度是(O(n))。所以对于每个数的判断,我们可接受的复杂度大约是(O(sqrt n))
考虑原来做过的题类似的做法。对于一个数(i),与这个数互质的数的本质实际上是不存在与(i)相同的质因子。所以对于每个(i),我们用(O(sqrt n))的复杂度求出其所有质因子作为预处理;
而对于每一个求出的质因子,任何含有该质因子的数都不能贡献到答案上。但是由于存在同时具有好几个该数质因子的情况,考虑容斥原理:
举个例子,如果一个数有3个质因子,设其为(a_1,a_2,a_3),则(ans=m-m/a1-m/a2-m/a3+m/a1/a2+m/a2/a3+m/a1/a3-m/a1/a2/a3)。其中$m是最大取值个数。
同理,设一个数有k个质因子,则容易知道:
(ans=m-m/(所有奇数个a_k相乘的可能)+m/(所有偶数个a_k)相乘的可能)
容易证明。当我们减去所有含有单个质因子的数时,我们多减去了所有严格含有两个质因子的数;再加上严格含有两个质因子的数,又多加了严格含有三个质因子的数;……以此类推,其实是(k)阶维恩图上的容斥原理。
所以我们可以用一个二进制数的每一位代表该质因子是否被选中。若选中了偶数个质因子,则加上;否则减去。
上代码:

#include<iostream>  #include<cstdio>  #include<cstring>  #include<cctype>  #include<cmath>  #define rep(i,a,n) for(int i=a;i<=n;i++)  #define dep(i,n,a) for(int i=n;i>=a;i--)  #define int long long  using namespace std;  int n,m,prime[100050],idx,ans,book_prime[100050],idx1;  bool is_prime[100050],book[100050],mark[100050];  inline int read()  {      int x=0,f=1;      char ch=getchar();      while(!isdigit(ch)){if(ch=='-')f=-1;ch=getchar();}      while(isdigit(ch))x=x*10+ch-'0',ch=getchar();      return x*f;  }  signed main()  {      memset(prime,-1,sizeof prime);      n=read(),m=read();      rep(i,1,n)      {          memset(book,0,sizeof book);          memset(mark,0,sizeof mark);          ans=m;          idx1=0;          int temp=i;          int qqqqqq=sqrt(i);          rep(j,2,qqqqqq)          {              if(temp%j==0)              {                  book_prime[++idx1]=j;                  while(temp%j==0)                      temp/=j;              }          }          if(temp>1)              book_prime[++idx1]=temp;          int maxn=(1<<idx1)-1;          rep(j,1,maxn)          {              int cnt=0;              int tmp=j;              int base=1;              int mul=1;              while(tmp)              {                  if(tmp&1)                      ++cnt,mul*=book_prime[base];                  ++base;                  tmp>>=1;              }              if(cnt&1)ans-=m/mul;              else ans+=m/mul;          }          printf("%lldn",ans);      }      return 0;  }