题目链接

题意 : 从[a,b]中找一个x,[c,d]中找一个y,要求GCD(x,y)= k。求满足这样条件的(x,y)的对数。(3,5)和(5,3)视为一组样例 。

思路 :要求满足GCD(x,y)=k的对数,则将b/k,d/k,然后求GCD(x,y)=1的对数即可。假设b/k >= d/k ;对于1到b/k中的某个数s,如果s<=d/k,则因为会有(x,y)和(y,x)这种会重复的情况,所以这时候的对数就是比s小的与s互质的数的个数,即s的欧拉函数。至于重复的情况是指:在d/k中可能有大于s的与d/k互质的数,但是当你把这些加上了之后,当你继续在b/k中找完s之后再找比s大的数的时候会产生和之前加上的情况重复的情况。

当s > d/k时,s分解质因数后,质因数的次数不影响结果。我们看另外那个区间有多少个和s不互质(减一下就好了),于是我们只要看另外那个区间中有多少个数是s质因数的倍数就好了。区间[1..a]中 p的倍数 显然有 a/p个。 我们枚举s的质因数利用容斥原理:看另外那个区间有多少个数与s不互质。

容斥原理的具体如下:

          区间中与s不互质的个数 = (区间中s的每个质因数的倍数个数)-(区间中s的每两个质因数乘积的倍数)+(区间中s的每3个质因数的成绩的倍数个数)-(区间中s的每4个质因数的乘积)+...

          于是问题变成了统计每个数的不同质因数的个数而忽略次数。这个可以用筛法。具体做法如下:

          对每个数保存一个真质因数的列表。初始每个列表的长度为0。然后从2开始,分别检查每个数的列表长度,如果列表长度不为0,则这个数是合数,跳过;如果这个长度为0,则我们找到了一个质数,同时再把这个数的倍数(不包含本身)的列表里加入这个数。

参考

 1 //1695
 2 #include <iostream>
 3 #include <string.h>
 4 #include <stdio.h>
 5 
 6 using namespace std ;
 7 
 8 int a,b,c,d,k,zh[100010][10],sh[100010] ;
 9 __int64 ans,eu[100010] ;
10 
11 void eular()
12 {
13     eu[1] = 1 ;
14     for(int i = 2 ; i < 100010 ; i++)
15     {
16         if(!eu[i])
17         {
18             for(int j = i ; j < 100010 ; j += i)
19             {
20                 if(!eu[j]) eu[j] = j ;
21                 eu[j] = eu[j] / i * (i-1) ;
22                 zh[j][sh[j]++] = i ;
23             }
24         }
25         eu[i] += eu[i-1] ;
26     }
27 }
28 __int64 dfs(int s ,int b,int i)
29 {
30     __int64 ans1 = 0 ;
31     for (int j = s ;j < sh[i] ; j++)
32     {
33         ans1 += b /zh[i][j] - dfs(j+1,b/zh[i][j],i);
34     }
35     return ans1;
36 }
37 int main()
38 {
39     int T ;
40     scanf("%d",&T) ;
41     int cas = 1 ;
42     eular() ;
43     while(T--)
44     {
45         scanf("%d %d %d %d %d",&a,&b,&c,&d,&k) ;
46         if(k == 0){
47             printf("Case %d: 0\n",cas++) ;
48             continue ;
49         }
50         int maxx = max(b,d) ;
51         int minn = min(b,d) ;
52         maxx /= k ;
53         minn /= k ;
54         ans = eu[minn] ;
55         for(int i = minn + 1 ; i <= maxx ; i++)
56             ans += minn - dfs(0,minn,i) ;
57         printf("Case %d: %I64d\n",cas++,ans) ;
58     }
59     return 0 ;
60 }
View Code

 

posted on 2014-05-23 21:39  枫、  阅读(329)  评论(0编辑  收藏  举报