Codeforces Round #511 (Div. 2) C. Enlarge GCD (质因数)

题目

题意:

  给你n个数a[1]...a[n],可以得到这n个数的最大公约数, 现在要求你在n个数中 尽量少删除数,使得被删之后的数组a的最大公约数比原来的大。 如果要删的数小于n,就输出要删的数的个数, 否则输出 -1 。

思路:

  设原来的最大公约数为 g, 然后a[1]...a[n]都除以g ,得到的新的a[1]...a[n],此时它们的最大公约数一定是1 。

  设除以g之后的数组a为:

              1    2    3     6      8   10

   则它们的质因数分别是:  1    2    3    2 3    2    2 5

  其中 质因数 2 的次数出现的最多,出现了4 次, 所以我们只要删除 n-4=2 个数就能使最大公约数由1 变成 2 。即删除 a[1]和a[3]就好,答案就是 2 。

  综上,只要找出质因数出现的最多的次数d, n-d就是我们要的答案。

  代码实现过程中,由于数据较大,要把筛质数 和 选因子  分开来做, 不能同时筛质因子(会超时),因为要避免筛质数这部分重复(打一次表就好)。  筛质数的时候,用2000以内的质数就够了(我也不知道为什么!)

  顺便一提:一个数m 的因子个数k 是小于log2m的 , 因为2^k<m 。 还有 数组至少能开1.5e7 大 。

 1 #include<iostream>
 2 #include<cstdio>
 3 #include <cctype>
 4 #include<algorithm>
 5 #include<cstring>
 6 #include<cmath>
 7 #include<string>
 8 #include<cmath>
 9 #include<set>
10 #include<vector>
11 #include<stack>
12 #include<queue>
13 #include<map>
14 using namespace std;
15 #define ll long long
16 #define mem(a,x) memset(a,x,sizeof(a))
17 #define se second
18 #define fi first
19 const ll mod=1e9+7;
20 const int INF= 0x3f3f3f3f;
21 const int N=3e5+5;
22
23
24 const int N2=1.5e7+5;
25 int n;
26 int cnt=0;
27 int check[2005];
28 int a[N];
29 int num[N2];
30 int prime[N];
31
32
33 int gcd(int x, int y)
34 {
35     return y==0?x:gcd(y,x%y);
36 }
37 void _prime()
38 {
39     int m=2000;
40     for(int i=2;i<=m;i++) //N以内的质数
41     {
42         if(!check[i])
43         {
44             prime[++cnt]=i;
45             for(int j=i;j<=m;j+=i)
46             {
47                 check[j]=1;
48             }
49         }
50     }
51 }
52 void factor(int m)
53 {
54     for(int i=1;i<=cnt;i++)
55     {
56         if(m%prime[i]==0)
57             num[prime[i] ]++;
58         while(m%prime[i]==0)
59         {
60             m/=prime[i];
61         }
62     }
63     if(m!=1) //包括了1
64         num[m]++;
65 }
66
67 int main()
68 {
69     cin>>n;
70     for(int i=1;i<=n;i++)
71         scanf("%d",&a[i]);
72
73     int g=a[1]; //g=删除前的最大公约数
74     for(int i=2;i<=n;i++)    g=gcd(g,a[i]);
75
76     _prime();
77     for(int i=1;i<=n;i++)
78     {
79         a[i]/=g;
80         factor(a[i]);
81     }
82
83     int ans=INF;
84     for(int i=1;i<=N2;i++)
85     {
86         if(num[i])
87             ans=min(ans,n-num[i]);
88     }
89     cout<< (ans<n? ans:-1 )<<endl;
90 }

原文地址:https://www.cnblogs.com/thunder-110/p/10122292.html

时间: 2024-11-05 00:42:04

Codeforces Round #511 (Div. 2) C. Enlarge GCD (质因数)的相关文章

Codeforces Round #511 (Div. 2)-C - Enlarge GCD (素数筛)

传送门:http://codeforces.com/contest/1047/problem/C 题意: 给定n个数,问最少要去掉几个数,使得剩下的数gcd 大于原来n个数的gcd值. 思路: 自己一开始想把每个数的因子都找出来,找到这些因子中出现次数最多且因子大于n个数的最大公约数的,(n - 次数 )就是答案.但是复杂度是1e9,差那么一点. 自己还是对素数筛理解的不够深.这道题可以枚举素数x,对于每个x,找到所有(a[i]/gcd(all)) 是x倍数的个数,就是一个次数.找这个次数的过程

Codeforces Round #511 (Div. 2) C. Enlarge GCD

题目链接 题目就是找每个数的最小素因子,然后递归除,本来没啥问题,结果今天又学习了个新坑点. 我交了题后,疯狂CE,我以为爆内存,结果是,我对全局数组赋值, 如果直接赋值,会直接在exe内产生内存,否则只会在运行时才分配内存. 1 #include <bits/stdc++.h> 2 using namespace std; 3 4 const int maxn = 1e7 + 5e6 + 10; 5 6 //线性素数筛 7 int prime[2000000],num_prime = 0;

Codeforces Round #511 (Div. 2)

又到了摸鱼的时候了23333 A. Little C Loves 3 I 题意:给一个数,分解为不被3整除的3个数 题解:构造,如果这个数被3整除,就构造为1,1,n-2:否则构造为1,2,n-3 1 class Solution(object): 2 def run(self): 3 n = int(input()) 4 if n % 3 == 0: 5 print(1, 1, n - 2) 6 else: 7 print(1, 2, n - 3) 8 9 if __name__ == '__

Codeforces Round #511 (Div. 1)

A - Enlarge GCD 题意:给n个数,那么他们有gcd,去掉最多n-1个数使得他们的gcd变大.求去掉最少的数. 题解:首先如果所有数都相等,那么无解.否则一定有解:最多去掉只剩下最大的那个.gcd是没有影响的,可以直接除掉(注意gcd可以用0来初始化,0和x的gcd都等于x).然后除去gcd之后每个数有他独特的几种因子,把不含这种因子的数都去掉就可以把这种因子释放出来.暴力sqrt分解会T掉,线性筛/埃筛预处理出每个数的最小质因子(甚至不需要预处理出他的幂,反正除一除也是log级别的

C. Enlarge GCD Codeforces Round #511 (Div. 2)【数学】

题目: Mr. F has nn positive integers, a1,a2,-,an. He thinks the greatest common divisor of these integers is too small. So he wants to enlarge it by removing some of the integers. But this problem is too simple for him, so he does not want to do it by

A. Little C Loves 3 I Codeforces Round #511 (Div. 2) 【数学】

题目: Little C loves number ?3? very much. He loves all things about it. Now he has a positive integer nn. He wants to split nn into 3 positive integers a,b,ca,b,c, such that a+b+c=na+b+c=n and none of the 3 integers is a multiple of 3. Help him to fin

Codeforces Round 511 Div.1 B

Description Given a \(n \times m\) chessboard, every time put two chessman with Manhattan distance 3 between them. Calculate the maximum number of chessmen you can put on it. \(n, m \le 10^9\). Solution All possible pairs of position is a biparite gr

Codeforces Round #428 (Div. 2)

Codeforces Round #428 (Div. 2) A    看懂题目意思就知道做了 #include<bits/stdc++.h> using namespace std; #pragma comment(linker, "/STACK:102400000,102400000") #define rep(i,a,b) for (int i=a; i<=b; ++i) #define per(i,b,a) for (int i=b; i>=a; --i

[Codeforces] Round #352 (Div. 2)

人生不止眼前的狗血,还有远方的狗带 A题B题一如既往的丝帛题 A题题意:询问按照12345678910111213...的顺序排列下去第n(n<=10^3)个数是多少 题解:打表,输出 1 #include<bits/stdc++.h> 2 using namespace std; 3 int dig[10],A[1005]; 4 int main(){ 5 int aa=0; 6 for(int i=1;;i++){ 7 int x=i,dd=0; 8 while(x)dig[++dd