求是第几个质数的简易算法
whqida2016/06/08软件综合 IP:湖北

昨天晚上看电影一不小心看到快4点,又一不小心打开科创,发现改版了,发现自己发帖要考试,还好过了,其中有一题,104667是第几个质数(不一定是这个数,记不太清了),我是用python编程求的

<code>def getPrime(n):
for i in range(2,n):
    if n%i == 0:
        return False
return True    

def  primePosition(x):
count = 0
if getPrime(x):
    for i in range(2, x):
        if getPrime(i):
            count += 1
    return count + 1
return count
</code>

结果是:9991 QQ截图12jpg.jpg

上面的函数不是很严谨,比如非法输入什么的都没处理。。。 差点忘了,话说python性能真不咋地,上面这题i5笔记本算了一分二十多秒。。。

[修改于 8年7个月前 - 2016/06/08 18:57:43]

来自:计算机科学 / 软件综合
28
已屏蔽 原因:{{ notice.reason }}已屏蔽
{{notice.noticeContent}}
~~空空如也
2480467935
8年7个月前 IP:山东
821171

打开Wolframalpha网站,然后搜搜就行了 注:国外网站,速度可能较慢。

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
whqida作者
8年7个月前 IP:湖北
821176

引用 2480467935 : 打开Wolframalpha网站,然后搜搜就行了 注:国外网站,速度可能较慢。

多谢提供的网站,果然够慢。验证了下,答案正确 413.jpg

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
bg8npk
8年7个月前 IP:广东
821205

blob.png

刚刚接触node.js,速度还行

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
nkc
8年7个月前 IP:广东
821209

于是,题库又少了一道题。。。

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
justinpiggy
8年7个月前 修改于 8年7个月前 IP:北京
821222

PrimePi[104677]
耗时不到2ms。。。
Screen Shot 2016-06-09 at 12.47.27 PM.png

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
acmilan
8年6个月前 修改于 8年6个月前 IP:四川
821330

VC++2015u2表示用楼主的算法在10秒以内出结果。。。

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
Cirno
8年6个月前 修改于 8年6个月前 IP:美国
821361

首先不敢苟同楼主对 Python 性能的评价,按楼主的标准,汇编才是最好的语言。

其次看到楼上不少人以换语言和 IDE跑同一个O(n2)算法而得到的运算时间优化来满足优越感,表示无法理解其中乐趣。

对于质数筛选,已经有很多前人做了很多算法效率上的优化,这里话不多说,给出链接 Sieve of Eratosthenes , Sieve of Atkin,talk is cheap.

这里用XXXXXX作为统一平台,消去了IDE环境和硬件配置的影响,来比较楼主的O(n2)方法和一种以空间换时间的方法的运行效率:

1.简单算法:

<code>int whichPrime(int num){
    if(num<4)return num-1; int numprime="2;" for(int i="4;i<=num;i++){" flag="1;" j="2;j*j<=i;j++){" if((i%j)="=0){" break; } numprime+="flag;" return numprime; < code></4)return></code>

测试结果 0.014957 S:
Screenshot from 2016-06-11 20:21:20.png

2.优化算法:

<code>int whichPrime(int num){
    if(num<4)return num-1; vector<int> list (num-1,1);
    for(int i=2;i*i<=num;i++){ if(list[i-2]){ for(int j="i*i;j<=num;j+=i){" list[j-2]="0;" } int numprime="0;" i="0;i<list.size();i++)" numprime+="list[i];" return numprime; < code></=num;i++){></4)return></code>

测试结果 0.001196 S:
Screenshot from 2016-06-11 19:57:07.png 快了十倍以上,当然这里并不是说用空间换时间的思路在任何情况下都可取,而且这个算法也远非最优解。只是提醒初学 coding 的 kcer,工具和语言在一定程度上都是浮云,“没有低级的法术,只有低级的法师”——《塔希里亚故事集》。

引用
评论
2
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
bg8npk
8年6个月前 IP:广东
821374

引用 琪露诺 : 首先不敢苟同楼主对 Python 性能的评价,按楼主的标准,汇编才是最好的语言。

其次看到楼上不少人以换语言和 IDE跑同一个O(n2)算法而得到的运算时间优化来满足优越感,表……

无形装逼,最为致命 【这很清真】

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
acmilan
8年6个月前 修改于 8年6个月前 IP:四川
821376

引用 琪露诺 : 首先不敢苟同楼主对 Python 性能的评价,按楼主的标准,汇编才是最好的语言。

其次看到楼上不少人以换语言和 IDE跑同一个O(n2)算法而得到的运算时间优化来满足优越感,表……

在VC2015u2上跑,

Release编译下,第一个是0.015089,第二个是0.000556,快了30倍

Debug编译下,第一个是0.024269,第二个是0.294831,慢了10倍以上

说明vector在Debug编译下由于要进行各种调试检查,资源和时间消耗非常大。

一个超(zuo)快(bi)的算法:

<code class="language-c">// 提前准备一个prime table
// 做binary search
</code>
引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
Cirno
8年6个月前 IP:美国
821382

引用 undefined : > 引用 琪露诺 : 首先不敢苟同楼主对 Python 性能的评价,按楼主的标准,汇编才是最好的语言。

其次看到楼上不少人以换语言和 IDE跑同一个O(n2)算法而得到……

回楼上,既然都用查表法了,为什么不更直接用 hash table 这种? O(1) 时间内就可以了。

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
acmilan
8年6个月前 修改于 8年6个月前 IP:四川
821391

引用 undefined : > 引用 undefined : > 引用 琪露诺 : 首先不敢苟同楼主对 Python 性能的评价,按楼主的标准,汇编才是最好的语言。

其次看到楼上不少人以换语言和 IDE跑同一个……

哈哈。。。还是你厉害。。。内存够的话,不如直接列个索引表了。。。

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
L834159672
8年6个月前 修改于 8年6个月前 IP:湖南
821429
<code class="language-python">from math import sqrt 

def getPrime(n):
    if (n!=2) & (n&1==0) :
        return False
    for i in range(3,int(sqrt(n))+1,2):
        if n%i == 0:
            return False
    return True   

def  primePosition(x):
    count = 0
    if x==2:
        return 1
    if (x>1) & getPrime(x):
        for i in range(3, x,2):
            if getPrime(i):
                count += 1
        return count + 2
    return count
</code>
引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
acmilan
8年6个月前 修改于 8年6个月前 IP:四川
821470

楼主的算法用C#跑的结果:

Debug x64版本,用了10.2742583秒

Release x64版本,用了7.5063185秒

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
cpp_programer
8年6个月前 IP:广东
822076
表示我是先列素数表再搜索的
引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
ComplexSG
8年0个月前 IP:江苏
828934
线性筛可破。O(N)
主要思想是,每个数只被他最小的质因数筛去。
从小到大看,如果某个数是质数,那就把这个数乘以所有比他最小的质因数小于或等于的质数得到的数全都筛掉,并且标记最小质因数。
嗯我说的好像不太清楚,这个网上全都是吧
引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
clk
7年3个月前 IP:陕西
839682
……只会c的弱鸡,还停留在函数调用,,,拿循环编出来的。比不过大佬们……
引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
精致大眼剑水蚤
5年0个月前 IP:广东
867298

用PyPy不好吗,加速后不还挺快的吗?

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
Leonard
5年0个月前 IP:浙江
867311

前面一位搞程序的说了,不是最优的算法,的确他算法和程序方面很大神了,但是我觉得炫耀成分多于想要帮人解决问题。

抛个数学为主的方法给你们,传送门:XXXXXXXXXXXXXXXXXXXXXXX/view/XXXXXXXXXXXXXXXXXXXXXXXXXXXml

这个复杂度怎么样?我不太会算,O(n)?

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
nullptr12
5年0个月前 修改于 5年0个月前 IP:山西
867373

楼上抛出的问题很有意思啊。


设函数ϕ(x) 等于不大于x的全体质数个数,则时间复杂度可以表示为O(ϕ(N)) 。

因为算法的关键路径:

用小于K的所有质数去除N

(其中K=N )

可通过一个循环体来实现,循环次数 T=ϕ(K1) 。

自然地,时间复杂度就为O(ϕ(N1)) = O(ϕ(N)) 。

PS: 这里还涉及一个数据循环依赖的问题,我们姑且认为程序事先知道小于K的所有质数。


那么问题来了,ϕ(x) 是个什么样的函数呢?似乎黎曼等学者曾经研究过相关问题,过于深奥,这就愁坏了我个非数学专业的。

不过,单看上界ϕ(x)x 必然成立,有O(ϕ(x))=O(x) ,于是得到时间复杂度为:

O(N) 。


虽然在全体正整数集上不清楚ϕ(x),但在有限范围内还是可以画出图像,观察得到N<=1e6总体呈线性增长,但斜率小于1。也就是说,这个算法可以作为同样具有O(N) 复杂度的暴力解法的系数优化版本。

批注 2019-12-13 181343.png

但是这个算法实用嘛?前面说过,存在数据的循环依赖,不利于计算机去算。如果打表的话,优势似乎就没了。。。

引用
评论(2)
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
Leonard
5年0个月前 IP:浙江
867404
引用nullptr12发表于26楼的内容
楼上抛出的问题很有意思啊。设函数ϕ(x) 等于不大于x的全体质数个数,则时间复杂度可以表示为O(ϕ(...

如果不查表的话,

用这种方法判断质数成功后,做标记,也是要比一个个判断效率高很多吧。


PS:用空间换时间后,这个算法效率还会更高。可惜我没楼上这么强的技术水平。只是一个想法

引用
评论
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论
nullptr12
5年0个月前 IP:山西
867406
引用Leonard发表于27楼的内容
如果不查表的话,用这种方法判断质数成功后,做标记,也是要比一个个判断效率高很多吧。PS:用空间换时间...

嗯嗯。但是做标记太笼统了。顺着链接的思路,您想说的应该是动态规划:从2开始,用这种方法由小到大逐个构建出素数表,但是每判断一个质数都要搜索前面的素数表(素数表用标记实现)。

其实换个方向就成了经典的埃氏素数筛:对于已经判断出是质数的数,不管它的前驱,而是标记后面已知不可能成为质数的数。这方法效率的确是高的,但是跟链接里面的算法还是有区别。


另外恳请管理员帮忙编辑一下,原文“楼上的抛出问题”改为“楼上抛出的问题”。

还有不是很懂为啥LaTex公式会渲染成这样。

引用
评论(1)
加载评论中,请稍候...
200字以内,仅用于支线交流,主线讨论请采用回复功能。
折叠评论

想参与大家的讨论?现在就 登录 或者 注册

所属专业
所属分类
上级专业
同级专业
文件下载
加载中...
{{errorInfo}}
{{downloadWarning}}
你在 {{downloadTime}} 下载过当前文件。
文件名称:{{resource.defaultFile.name}}
下载次数:{{resource.hits}}
上传用户:{{uploader.username}}
所需积分:{{costScores}},{{holdScores}}下载当前附件免费{{description}}
积分不足,去充值
文件已丢失

当前账号的附件下载数量限制如下:
时段 个数
{{f.startingTime}}点 - {{f.endTime}}点 {{f.fileCount}}
视频暂不能访问,请登录试试
仅供内部学术交流或培训使用,请先保存到本地。本内容不代表科创观点,未经原作者同意,请勿转载。
音频暂不能访问,请登录试试
支持的图片格式:jpg, jpeg, png
插入公式
评论控制
加载中...
文号:{{pid}}
投诉或举报
加载中...
{{tip}}
请选择违规类型:
{{reason.type}}

空空如也

加载中...
详情
详情
推送到专栏从专栏移除
设为匿名取消匿名
查看作者
回复
只看作者
加入收藏取消收藏
收藏
取消收藏
折叠回复
置顶取消置顶
评学术分
鼓励
设为精选取消精选
管理提醒
编辑
通过审核
评论控制
退修或删除
历史版本
违规记录
投诉或举报
加入黑名单移除黑名单
查看IP
{{format('YYYY/MM/DD HH:mm:ss', toc)}}