该楼层疑似违规已被系统折叠 隐藏此楼查看此楼

[原创]求质数(C语言描述)

【问题描述】:

试编写一个程序,找出2->N之间的所有质数。希望用尽可能快的方法实现。

【问题分析】:

这个问题可以有两种解法:一种是用“筛子法”,另一种是从2->N检查,找出质数。

先来简单介绍一下“筛法”,求2~20的质数,它的做法是先把2~20这些数一字排开:

2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20

先取出数组中最小的数,是2,则判断2是质数,把后面2的倍数全部删掉。

2 | 3  5 7 9 11 13 15 17 19

接下来的最小数是3,取出,再删掉3的倍数

2 3 | 5 7 11 13 17 19

一直这样下去,直到结束。

筛法求质数的问题时,非质数的数据有很多是重复的。例如,如果有一个数3×7×17×23,那

么在删除3的倍数时会删到它,删7、17、23时同样也会删到它。有一种“线性筛法”,可以安排删除的

次序,使得每一个非质数都只被删除一次。从而提高效率。因为“筛法”不是我要介绍的重点,所以就

不介绍了。

现在我来介绍第二种方法。用这种方法,最先想到的就是让从2~N逐一检查。如果是就显示出来

,如果不是,就检查下一个。这是正确的做法,但效率却不高。当然,2是质数,那么2的倍数就不是质

数,如果令i从2到N,就很冤枉地测试了4、6、8……这些数?所以第一点改建就是只测试2与所有的奇数

就足够了。同理,3是质数,但6、9、12……这些3的倍数却不是,因此,如果能够把2与3的倍数跳过去

而不测试,任意连续的6个数中,就只会测试2个而已。以6n,6n+1,6n+2,6n+3,6n+4,6n+5为例,

6n,6n+2,6n+4是偶数,又6n+3是3的倍数,所以如果2与3的倍数都不理会,只要测试的数就只留下6n+1和

6n+5而已了,因而工作量只是前面想法的2/6=1/3,应该用这个方法编程。

还有个问题,就是如果判断一个数i是否为素数。按素数的定义,也就是只有1与本身可以整除

,所以可以用2~i-1去除i,如果都除不尽,i就是素数。观点对,但却与上一点一样的笨拙。当i>2时,

有哪一个数可以被i-1除尽的?没有,为什么?如果i不是质数,那么i=a×b,此地a与b既不是i又不是1

;正因为a>1,a至少为2,因此b最多也是i/2而已,去除i的数用不着是2~i-1,而用2~i/2就可以了。不

但如此,因为i=a×b,a与b不能大于sqrt(i),为什么呢?如果a>sqrt(i),b>sqrt(i),于是a×b>sqrt(i)

*sqrt(i)=i,因此就都不能整除i了。如果i不是质数,它的因子最大就是sqrt(i);换言之,用2~sqrt

(i)去检验就行了。

但是,用2~sqrt(i)去检验也是浪费。就像前面一样,2除不尽,2的倍数也除不尽;同理,3除

不尽,3的倍数也除不尽……最理想的方法就是用质数去除i。

但问题是这些素数从何而来?这比较简单,可以准备一个数组prime[],用来存放找到的素数,

一开始它里面有2、3、5。检查的时候,就用prime[]中小于sqrt(i)的数去除i即可,如果都除不尽,i就

是素数,把它放如prime[]中,因此prime[]中的素数会越来越多,直到满足个数为止!

不妨用这段说明来编写这个程序,但是程序设计的时候会有两个小问题:

1.如果只检查6n+1和6n+5?不难发现,它们的距离是4、2、4、2……所以,可以先定义一个变

量gab=4,然后gab=6-gab;

2.比较是不能用sqrt(i),因为它不精确。举个例子,i=121,在数学上,sqrt(i)自然是11,但

计算机里的结果可能是10.9999999,于是去除的数就是2、3、5、7,而不含11,因此121就变成质数了。

解决这个问题的方法很简单,不要用开方,用平方即可。例如,如果p*p<=i,则就用p去除i。而且它的

效率比开方高。

【程序清单】:

#include

int creat_prime(int prime[],int n,int total)

{

register        int        i;

register        int        j;

register        int        gab=2;

register        int        count;

for(i=7;i<=n;i+=gab)

{

count=1;

gab=6-gab;

for(j=0;prime[j]*prime[j]<=i;j++)

{

if(i%prime[j]==0)

{

count=0;

break;

}

}

if(count)

{

prime[total]=i;

total++;

}

}

return total;

}

int main(void)

{

int        prime[30000]={2,3,5};

int        total=3;        //找到素数的个数

int        i;

int        n=200000;        //要查找的范围(>=6)

total=creat_prime(prime,n,total);

for(i=0;i

{

printf("%d ",prime[i]);

if(i && !(i%10))

putchar('\n');

}

putchar('\n');

}

如果你有VC++或其他C编译器,可以去运行一下。

你会发现,会有很多时间去输出。我给个小提示,

假设你编译生成的可执行文件是prime.exe,目录在D:\

那你在命令行下进如D:\然后输入命令 prime >1.txt

这样就可以把结果保存到1.txt。

你会发现在int越界的前提下,它几乎都是瞬间完成的!

当然这段程序还是有可以改进的地方,如果你有更好的建议,请联系我。

redraiment@126.com

C语言找n以上最小质数,[原创]求质数(C语言描述)相关推荐

  1. python中求质数_python求质数的3种方法

    本文为大家分享了多种方法求质数python实现代码,供大家参考,具体内容如下 题目要求是求所有小于n的质数的个数. 求质数方法1: 穷举法: 根据定义循环判断该数除以比他小的每个自然数(大于1),如果 ...

  2. c语言找出比n小的最大质数,C++ 实现求小于n的最大素数的实例

    C++ 实现求小于n的最大素数的实例 枚举就是基于已有知识镜像答案猜测的一种问题求解策略 问题:求小于n的最大素数 分析: 找不到一个数学公式,使得根据N就可以计算出这个素数 我们思考: N-1是素数 ...

  3. c语言汇编混合编程写一个乘法,求通过C语言实现矩阵的加、减及乘法。要自己写的,不要复制过来...

    满意答案 eevfikx2 2013.11.28 采纳率:53%    等级:13 已帮助:8891人 #include using namespace std; int main() { int a ...

  4. c语言已知三点的坐标,求一个C语言算法:已知三角形的三个顶点的坐标,求三角形面积...

    共回答了16个问题采纳率:87.5% //三个顶点坐标是6个参数. #include #include double S_triangle(double ax,double ay,double bx, ...

  5. c语言整形变量x 进制转换,求一个C语言程序 输入整型变量x 输出整型变量x对应的十进制、八进制和十六进制形...

    满意答案 发的地方ssd发 推荐于 2018.05.10 采纳率:42%    等级:9 已帮助:112人 printf自带输出格式. %d是十进制输出, %x是十六进制输出. %o是八进制输出.用p ...

  6. C语言找出数组中最小的数和它的下标

    C语言找出数组中最小的数和它的下标,然后把它和数组中最前面的元素对换位置 #include <stdio.h> int main() {int A[10]={3,7,5,9,10,2,1, ...

  7. c语言找出100内孪生质数,100以内所有孪生质数

    using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.T ...

  8. c语言求13为质数的代码,C语言求质数.doc

    C语言求质数 试编写一个程序,找出2->N之间的所有质数.希望用尽可能快的方法实现.[问题分析]: 这个问题可以有两种解法:一种是用"筛子法",另一种是从2->N检查, ...

  9. 质数c语言欧拉筛选,Python|欧拉筛法求质数

    欢迎点击「算法与编程之美」↑关注我们! 本文首发于微信公众号:"算法与编程之美",欢迎关注,及时了解更多此系列文章. 问题描述 我们知道第一个质数是 2.第二个质数是 3.第三个质 ...

  10. C语言---找零问题------程序优化

    C语言----找零问题----程序优化 题目详情: 现有一元,两元,五元面额的纸币若干,给出任意的找零要求,给出每种找零方案,输出循环次数. 优化代码一: 代码优化内容: 利用num/5,num/2来 ...

最新文章

  1. 2017年智能家居将从概念走进现实
  2. armv8 汇编 绝对地址赋值_ARMv8带来的变化
  3. PaddleHub 视频人脸检测
  4. Qt+OpenCV界面
  5. python如何下载zip文件_使用python-mechaniz下载zip文件
  6. 阿里巴巴整理的python_阿里P8大佬整理的2020年最全99道python面试题,文末附答案...
  7. linux漏洞知乎_Linux本地内核提权漏洞(CVE-2019-13272)
  8. 哪个服务器比较稳定,云服务器哪个比较稳定
  9. prim算法适用条件_内部排序算法的比较及应用
  10. 酷Q移植先驱的记录流程,QQ机器人框架插件
  11. OkHttp之BridgeInterceptor简单分析
  12. First non-repeating character
  13. postgresql数据库中多个Schemas互相访问
  14. Mac最新版书籍分享
  15. unity开发VR,没有VR设备解决方式
  16. 2012-10-19《商业英语中的加班表达…
  17. 了解寄存器: EBP寄存器
  18. 我知道你不想跳槽,但你不该拒绝面试机会
  19. 整除/ 与 求余% 运算
  20. Java设置mysql字段为null_MySQL字段的属性应该尽量设置为NOT NULL

热门文章

  1. 平面设计师okr_为什么说一名优秀的设计师要懂得OKR
  2. vSphere Esxi 7.0打包网卡驱动
  3. python学习笔记30(利用turtle绘制最基本的几何图形)
  4. 餐厅点餐系统设计思路
  5. 通用汽车:导航地图的未来是基于区块链的
  6. NLP基础--single-pass 聚类算法
  7. MacBook Air密码忘了,苹果电脑密码忘了怎么办
  8. 【lidar】单目深度估计与伪雷达点云、可视化
  9. 使用Hbase Shell和Java API分别进行Hbase的增删改查操作
  10. java:记一次因时区导致出参和入参时差8小时前因后果