何为"打表"呢,说得简单点就是:
有时候与其重复运行同样的算法得出答案,还不如直接用算法把这组数据所有可能的答案都枚举出来存到一个足够大的容器中去-例如数组(打表),然后再输入数据的时候,直接遍历容器,检索这个数据是否有题意要求的结果。
这一题大意是给出 多组N(1~1000)和C,让你从N内素数的中间项向外扩展C个素数,比如给出7 1,素数有5个(注意此题出题人坑爹得让1作为"素数") 1,2,3,5,7,那么应输出7 1:3 5 7(注意输出格式)
特别的,偶数个素数输出2c-1个,奇数个则输出2c个。
那么有几点我们是需要分析的:
第二,三点是细节问题,但是第一点则是算法优化问题了,常见的素数算法优化就是:打表法 and 筛选法。
Code附上:
1 //素数打表(筛选法) 2 //Memory: 140 K,Time: 0 Ms 3 #include<iostream> 4 #include<cstdio> 5 #include<cmath> 6 using namespace std; 7 8 #define MAX 1005 9 10 int prime[MAX],pl; 11 int num[MAX]; 12 13 /*筛选法素数打表*/ 14 void prime_number(int max_num) 15 { 16 int i,j; 17 int k = (int)sqrt(1.0*max_num); //开方-四舍五入 18 19 /*标记所有合数*/ 20 for(i=2;i <= k;i++) 21 { 22 if(!num[i]) 23 for(j = 2*i; j<=max_num; j+=i) 24 num[j] = 1; 25 } 26 27 /*打表*/ 28 pl = 1; //素数计数器 29 for(i=1;i <= max_num;i++) //ps:此题"素数"包括1 30 { 31 if(!num[i]) 32 prime[pl++] = i; 33 } 34 return; 35 } 36 37 int main() 38 { 39 int n,c; 40 int i; 41 42 prime_number(MAX); //先打好最大素数表 43 44 while(~scanf("%d%d",&n,&c)) 45 { 46 /*找到素数区间*/ 47 for(i = 1; i < pl; i++) 48 { 49 if(n < prime[i]) 50 break; 51 } 52 int maxp = i-1; 53 54 printf("%d %d:",n,c); 55 if(maxp%2) //奇数个 56 { 57 int mid = (maxp+1)/2; 58 if(2*c-1 > maxp) 59 c = mid; 60 61 for(i = mid-c+1; i <= mid+c-1; i++) 62 printf(" %d",prime[i]); 63 } 64 else //偶数个 65 { 66 int mid = maxp/2; 67 if(2*c > maxp) 68 c = mid; 69 70 for(i = mid-c+1;i <= mid+c; i++) 71 printf(" %d",prime[i]); 72 } 73 printf("\n\n"); 74 } 75 76 return 0; 77 }
Loading...
ACM/ICPC 算法训练 之 "打表"思路(防超时) ——附加素数筛选法
原文:http://www.cnblogs.com/Inkblots/p/4715387.html