• 最大公约数——Hankson的趣味题(线筛法求质数+gcd+质因数组合搜索约数)


    传送门:200. Hankson的趣味题 - AcWing题库

    思路:题目中给定的条件是gcd(a,x)=a1, lcm(b,x)=b1;

    容易发现x一定是b1的约数,所以可以尝试求出b1的所有约数看一下是否满足上面两个条件。

    1.试除法求约数,题目多测试样例,时间复杂度为O(n * √b1 * logb1),经测试,以下代码有一个样例过不了。

    1. #include
    2. #include
    3. #include
    4. using namespace std;
    5. typedef pair<int,int>PII;
    6. typedef long long ll;
    7. const int N=1e5;
    8. int factor[N],cnt;
    9. int gcd(int a,int b)
    10. {
    11. return b?gcd(b,a%b):a;
    12. }
    13. int main()
    14. {
    15. int n;
    16. cin>>n;
    17. while(n--)
    18. {
    19. int a,b,a1,b1;
    20. cin>>a>>a1>>b>>b1;
    21. int d=b1;
    22. cnt=0;
    23. for(int i=1;i*i<=d;i++)
    24. {
    25. if(d%i==0)
    26. {
    27. factor[++cnt]=i;
    28. if(i!=d/i)
    29. factor[++cnt]=d/i;
    30. }
    31. }
    32. int sum=0;
    33. for(int i=1;i<=cnt;i++)
    34. {
    35. int x=factor[i];
    36. if(gcd(x,a)==a1&&(ll)b*x/gcd(b,x)==b1)
    37. sum++;
    38. }
    39. cout<
    40. }
    41. return 0;
    42. }

    2.预处理出1~√2e9的所有质数,求出b1的所有质因数及其个数,其中质因数的个数不会超过9个,

    2*3*5*7*11*13*17*19*23*29=6469693230为60多亿,远大于题目给的20亿。再在质因数中搜索所有的约数。

    约数——正约数个数求法及其原理,求N的正约数集合_北岭山脚鼠鼠的博客-CSDN博客

    中试除法推论可知,一个数的约数个数最多为2*(√N)

    但实际上,一个数的约数个数远没有这么多,20亿以内约数个数最多的数的约数个数为1536。

     因此搜索约数的次数实际很少。

    第一步:预处理1~√2e9的所有质数,时间复杂度为O(√2e9),

    第二步:在质数数组中找出b1的所有质因数。

    第三步:搜索约数。

    第四步:遍历约数数组看是否满足两个条件。

    1. #include
    2. #include
    3. #include
    4. using namespace std;
    5. typedef pair<int,int>PII;
    6. typedef long long ll;
    7. const int N=1e5;
    8. int primes[N],cntp;
    9. PII factor[2000];
    10. bool st[N];
    11. int cntf;
    12. int d[N],cntd;
    13. void dfs(int u,int p)//u是当前找到了第几个约数,p是当前的约数大小。
    14. {
    15. if(u>cntf)
    16. {
    17. d[++cntd]=p;
    18. return ;
    19. }
    20. for(int i=0;i<=factor[u].second;i++)
    21. {
    22. dfs(u+1,p);
    23. p*=factor[u].first;
    24. }
    25. }
    26. void get_primes(int n)
    27. {
    28. for(int i=2;i<=n;i++)
    29. {
    30. if(!st[i]) primes[++cntp]=i;
    31. for(int j=1;primes[j]<=n/i;j++)
    32. {
    33. st[primes[j]*i]=true;
    34. if(i%primes[j]==0) break;
    35. }
    36. }
    37. }
    38. int gcd(int a,int b)
    39. {
    40. return b?gcd(b,a%b):a;
    41. }
    42. int main()
    43. {
    44. get_primes(N);
    45. int n;
    46. cin>>n;
    47. while(n--)
    48. {
    49. int a,b,a1,b1;
    50. cin>>a>>a1>>b>>b1;
    51. int d1=b1;
    52. cntf=0;
    53. for(int i=1;primes[i]<=d1/primes[i];i++)
    54. {
    55. if(d1%primes[i]==0)
    56. {
    57. int s=0;
    58. while(d1%primes[i]==0) s++,d1/=primes[i];
    59. factor[++cntf]={primes[i],s};
    60. }
    61. }
    62. if(d1>1) factor[++cntf]={d1,1};
    63. cntd=0;
    64. int sum=0;
    65. dfs(1,1);
    66. for(int i=1;i<=cntd;i++)
    67. {
    68. int x=d[i];
    69. if(gcd(x,a)==a1&&(ll)b*x/gcd(b,x)==b1)
    70. sum++;
    71. }
    72. cout<
    73. }
    74. return 0;
    75. }
  • 相关阅读:
    2D游戏案例:游戏场景搭建
    数字化时代,数据就是资产
    ESMapping字段
    [网络工程师]-传输层协议-TCP拥塞控制
    会计学基础期末考试试题及答案
    牛客每日刷题之二叉树
    RabbitMQ之延迟队列
    [buuctf][ACTF新生赛2020]usualCrypt
    Linux系统编程 系统编程概念
    汽车行业分论坛 | 让数据行驶在“安全道”
  • 原文地址:https://blog.csdn.net/m0_62327332/article/details/126673446