这题不知wA多小次了,先是打素数表超时,后是取余数超时。
没法,在绝望的情况下看了大牛的,我无语了。。。。。
题意:(英语基础不行,题意都很坑爹)。
给你一个数,它是由两个素数相乘而得的,再给你一个数M,如果其中有一个素数小于该数M,就输出“BAD”,并且要输出该素数。只有两个素数都它大于这数M,才输出“GOOD”。
(这题目废话真多!,何必呢?)
分析题目:
一看到素数,就想到了打素数表,这是解决这题的第一步,但怎样快速呢,又是一个麻烦事。另外,平时我们用的数都是十进制的数。所以,这题就要坑你,人是活的,想想看,
为什么又二进制,八进制,十六进制。还不是处理方便些。计算机有点笨,它只认识1和0,如果数字很大我们这样处理呢,既然有这么多进制,并且他有规律,就是处理为位。比如
十六进制中字母可以代表两位数,100进制代表三位数。如果1000进制,那它一位可代表四位数了。记住,这就有点像数小数点,一千是数三位的。它本身要保留一位,这可
别搞错了。
素数筛选:
一个合数都至少含有一个素因子。这是素数筛选的思路。比如 30=2*3*5,它含有三个,我们只要用2去筛选就行了,没必要用3和5再去筛选。所以它筛选的速度比较快
记住:不能用老土的方法去筛选,会TLE的。
同余定理:
比如:(a+b)%c=(a%c+b%c)%c; 相加取余(相减也类式)
(a*b)%c=((a%c)*(b%c))%c; 相乘取余
这个可以扩展到多个数相乘或相加。
明白了这些,这题就好办了。
参考代码:(四位为什么WA!)三位却AC了。
#include<stdio.h> #include<string.h> #define L 1000010 char a[105]; int kt[10000],k,prime[L]; int save(int h) { int sum=0; for(int i=k-1;i>=0;i--) { //sum=(sum*10000+kt[i])%h; sum=(sum*1000+kt[i])%h;//递归取余。 } return sum; } int main() { int flag,b,i,m=1; prime[0]=2; for(i=3;i<L;i+=2) //素数 筛选 { flag=1; for(int j=0;prime[j]*prime[j]<=i;j++) if(i%prime[j]==0) { flag=0; break; } if(flag) prime[m++]=i; } while(scanf("%s%d",a,&b),b) { memset(kt,0,sizeof(kt)); k=strlen(a); flag=1; for(int j=0;j<k;j++) { int r=(k+2-j)/3-1; //不足三位的,暂时补足,减1代表了它是否超过三位。 //int r=(k+3-j)/4-1; kt[r]=kt[r]*10+(a[j]-'0'); } k=(k+2)/3; //k=(k+3)/4; for(i=0;prime[i]<b;i++) { if(!save(prime[i])) { flag=0; break; } } if(flag) printf("GOOD\n"); else printf("BAD %d\n",prime[i]); } return 0; }
筛选方法一: 其中 MAX=1000100 ,p[MAX];
for (int i = 4; i <= MAXN; i += 2) { p[i] = 1; } int LIM = (int)sqrt(1.0*MAXN); for (int i = 3; i <= LIM; i += 2) { if (p[i]) continue; int k = 2 * i; for (int j = i*i; j <= MAXN; j += k) { p[j] = 1; } } for (int i = 2; i <= MAXN; ++i) { if (!p[i]) rec[++idx] = i; } }
筛选方法二:线性筛选
void getprime() { idx = -1; for (int i = 2; i <= MAXN; ++i) { if (!p[i]) { rec[++idx] = i; } for (int j = 0; rec[j]*i <= MAXN; ++j) { p[rec[j]*i] = 1; if (i % rec[j] == 0) { break; } } }