记一次【求n以内的素数个数】的优化记录
最近在leetCode上刷提,还是满锻炼人的,为以后面试打基础吧。不多说下面开始。
问题:求[2,n]之间的素数的个数。
来源:leetCode OJ
提示:
Let's start with a isPrime function. To determine if a number is prime, we need to check if it is not divisible by any number less than n. The runtime complexity of isPrime function would be O(n) and hence counting the total prime numbers up to n would be O(n2). Could we do better?
先让我们写一个函数isPrime用于判断给定的数是否是素数,为了判别总一个数是否是素数,我们需要依次检查比这个数小的数能否整除n。那么函数isPrime的时间复杂度就将是O(n),因此总体的时间复杂度就会达到O(n2),我们能做的更好吗?
第一次尝试
预备知识
① 什么是素数:素数(又叫质数),与之相反的是合数。如果只考虑非负数范围内,一个大于1的数,它只能被 1 和 他本身整除。这样的数就是素数。
② 0 和 1既不是质数也不是合数。
③ 素数定理:如果一个数x是素数,那么在整数范围[2,√x ]之间,找不到任何能整除x 的整数。因此,我们不必对 [2, n) 的所有整数去尝试,而只需要对 [2,√x]之间的数尝试整除就OK了,节约了时间。
#include<iostream>
#include<cassert>
#include<time.h>
#include<cmath>
using namespace std; /**
作用:判断一个数是否是素数
参数x:待判断的素
返回:是素数返回true,否则返回false
*/
bool isPrime(int x)
{
for(int i=;i<=int(sqrt(x));++i)
{
if(x%i==) return false;
}
return true;
} /**
作用:统计[2,n]之间的素数的个数
参数:n
返回:素数的个数
*/
int countPrimes(int n) {
int count=; if(n<) return ; for(int i=;i<=n;++i)
{
if(isPrime(i)) //如果 i为素数
{
++count;
}
}
return count; } int main()
{ clock_t start = clock();
int total = countPrimes();
clock_t end = clock(); cout<<"耗时:"<<(double(end-start))/CLOCKS_PER_SEC*<<"毫秒"<<endl; return ;
} /**********测试数据**************************
n=200000 81毫秒
n=700000 385毫秒
*/
这是很多人第一感觉写出来的solution,它最大问题在于使用了sqrt函数。第一:sqrt是用来处理浮点数的,而浮点数的计算速度远远慢于integer。第二,函数调用也会造成时间的浪费。第三:浮点数的存储误差可能引出致命错误,如 sqrt(9) 可能等于 2.9999999 ,那么 int(sqrt(9)) 就等于2 而不是3。
但是我从来都不是这样写的,这个让我印象很深,在我的编程启蒙书 《C prime Plus》书中,它教我使用 C/C++内置于语言的运算符乘 * ,而不是sqrt()函数。
第二次尝试
#include<iostream>
#include<cassert>
#include<time.h>
#include<cmath>
using namespace std; /**
作用:判断一个数是否是素数
参数x:待判断的素
返回:是素数返回true,否则返回false
*/
bool isPrime(int x)
{
for(int i=;i*i<=x;++i) //改进:使用 i*i<=x 而不是 i<= sqrt(x)
{
if(x%i==) return false;
}
return true;
} /**
作用:统计[2,n]之间的素数的个数
参数:n
返回:素数的个数
*/
int countPrimes(int n) {
int count=; if(n<) return ; for(int i=;i<=n;++i)
{
if(isPrime(i)) //如果 i为素数
{
++count;
}
}
return count; } int main()
{ clock_t start = clock();
int total = countPrimes();
clock_t end = clock(); cout<<"耗时:"<<(double(end-start))/CLOCKS_PER_SEC*<<"毫秒"<<endl; return ;
} /*********************测试数据*****************************
n=200000 46毫秒
n=700000 194毫秒
*/
可以发现,小小的改变,时间复杂度减少了大半!!!让我觉得很遗憾的是,很少有人知道这个技巧,leetCode上,也没有发现使用这个技巧的。
第三次尝试
前面的2个solution是不能Accept的,时间复杂度不达标。下面我又开始了google,找到了如下的优化方案。
预备知识:
① 合数一定能分解为 若干个 质素相乘 。如 27 = 3x3x3 ,155 = 5x31
② 原命题的真假性 和他的逆否命题相同。因此我们可以推导出第①命题的逆否命题--->: 不能分解为 素数相乘的数一定是质素(素数)。
因此根据这个推导出的定理,我们不再需要将待判断的所有数x去对 [2,sqrt(x)]之间的数试除,而只需要去对 [2,sqrt(x)]之间的素数试除就OK了。所以:我们必须在迭代判断素数的过程中,将已经判定为素数的数用数组存储起来,因为后面更大的数判断时,需要用比他小的素数(在[2,sqrt(x)]之间的)去试除做判断。那么这个数组需要多大呢?
[0,n]之间有 n +1 个整数,然而0 和 1不是素数也不是质素,因此剩下 n +1 -2 = n-1 个数。素数偶数各一半,除了2,偶数一定不是素数,因此我们大致将这个数组大小定义为 (n-1)/2 +1
好了,时间 和 空间上都过了优化,下面试试吧。
#include<iostream>
#include<cassert>
#include<time.h>
#include<cmath>
using namespace std; /**
作用:判断一个数是否是素数
参数x:待判断的素
参会primes:存储比x小的所有素数 的数组
返回:是素数返回true,否则返回false
*/
bool isPrime(int x,int *primes)
{
for(int i=;primes[i]*primes[i]<=x;++i) //用primes中存储的素数做为试除因子。
{
if(x%primes[i]==) return false;
}
return true;
} /**
作用:统计[2,n]之间的素数的个数
参数:n
返回:素数的个数
*/
int countPrimes(int n) { if(n<) return ; int count=;
int* primes = new int[(n-)/+] ; if(n >=) primes[count++] = ; //2是第一个素数,直接将他放入数组。 for(int i=;i<=n;++i)
{
if(isPrime(i,primes)) //如果 i为素数
{
primes[count++] = i; //则将 i存储到数组中
}
}
delete[] primes;
return count; } int main()
{ clock_t start = clock();
int total = countPrimes();
clock_t end = clock(); cout<<"耗时:"<<(double(end-start))/CLOCKS_PER_SEC*<<"毫秒"<<endl; return ;
} /****************测试数据******************
n=200000 13毫秒
n=700000 50毫秒
*/
时间复杂度又大大减少了!!!可见,我们的程序优化的空间往往很大。
最后,我的程序终于被AC了 ,不过,还可以继续优化,下面是我的程序的所处的时间复杂度的位置,前面有70% 的solution比我更优 。优化尚未成功,码农仍需努力。
:)

更佳的solution 请看这篇
欢迎转载,请注明出处:www.cnblogs.com/lulipro
为了获得更好的阅读体验,请访问原博客地址。
限于本人水平,如果文章和代码有表述不当之处,还请不吝赐教。
代码钢琴家
记一次【求n以内的素数个数】的优化记录的更多相关文章
- 【题目】求n以内的素数个数
最近在leetCode上刷提,还是满锻炼人的,为以后面试打基础吧.不多说下面开始. 问题:求[2,n]之间的素数的个数. 来源:leetCode OJ 提示: Let's start with a i ...
- Python练习题 026:求100以内的素数
[Python练习题 026] 求100以内的素数. ------------------------------------------------- 奇怪,求解素数的题,之前不是做过了吗?难道是想 ...
- Python3求m以内的素数、求m个数中最小的n个数
[本文出自天外归云的博客园] 题1:求m以内的素数(m>2) def find_all_primes_in(m): def prime(num): for i in range(2, num): ...
- 【C语言】输入一个整数N,求N以内的素数之和
[C语言]输入一个整数N,求N以内的素数之和 /* ========================================================================== ...
- 斐波那契数列(递归)&求100以内的素数
Java 5 添加了 java.util.Scanner 类,这是一个用于扫描输入文本的新的实用程序.它是以 前的 StringTokenizer 和 Matcher 类之间的某种结合.由于任何数据都 ...
- Algorithm --> 求N以内的真分数个数
求N以内的真分数个数 For example, if N = 5, the number of possible irreducible fractions are 11 as below. 0 1/ ...
- 求1e11以内的素数
有两种做法,一种是打表,另一种是直接求. 打表 将1e11每隔len(len=2000w)个数字统计一下该区间内素数的个数,比如cnt[1] 表示[1,len]以内有多少个素数,cnt[2]表示[le ...
- 求小于n的素数个数
本文是对 LeetCode Count Primes 解法的探讨. 题目: Count the number of prime numbers less than a non-negative num ...
- python 求100以内所有素数
def prime(num): for i in range(2, num): if num % i == 0: # 能被1之外的任意个数整除的即为非素数,返回False,将被filter函数过滤掉 ...
随机推荐
- 【linux】su、sudo、sudo su、sudo -i的用法和区别
来源:http://bbs.csdn.net/topics/390938651 sudo : 暂时切换到超级用户模式以执行超级用户权限,提示输入密码时该密码为当前用户的密码,而不是超级账户的密码.不过 ...
- myeclipse6.5注册机
import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; public ...
- jwplayer播放器停止 单页内多个jwplayer对象停止问题
单页内多个jwplayer对象停止问题,一直没有找到单页内多个jwplayer播放器停止问题,点击其中一个停止其他播放器; 整个播放代码Remove(),这样就可以停止了,也在ie下防止暂停不了.重音 ...
- PHP日期与时间
时间戳是自 1970 年 1 月 1 日(00:00:00 GMT)以来的秒数.它也被称为 Unix 时间戳(Unix Timestamp).Unix时间戳(Unix timestamp),或称Uni ...
- VR技术的探索阶段
转载请声明转载地址:http://www.cnblogs.com/Rodolfo/,违者必究. 早在1929年,在长期使用教练机训练器(机翼变短,不能产生离开地面所需的足够提升力)进行飞行训练之后,E ...
- Angular.element和$document的使用方法分析,代替jquery
AngularJs是不直接操作DOM的,但是在平时的开发当中,我们有的时候还是需要操作一些DOM的,如果使用原生的JS的话操作过于麻烦,所以大家一般都是使用jQuery,jQuery虽然好用,但是An ...
- JVM 1.6 GC
JVM调优是一门艺术. JVM调优的重点是减少Major GC的次数,因为Major GC会暂停程序比较长的时间.如果Major GC的次数比较多,意味着应用程序的JVM内存参数需要调整. JVM内存 ...
- java线程池ThreadPoolExecutor使用简介
一.简介线程池类为 java.util.concurrent.ThreadPoolExecutor,常用构造方法为:ThreadPoolExecutor(int corePoolSize, int m ...
- BZOJ 3196 Tyvj 1730 二逼平衡树 ——树状数组套主席树
[题目分析] 听说是树套树.(雾) 怒写树状数组套主席树,然后就Rank1了.23333 单点修改,区间查询+k大数查询=树状数组套主席树. [代码] #include <cstdio> ...
- wpf图片查看器,支持鼠标滚动缩放拖拽
最近项目需要,要用到一个图片查看器,类似于windows自带的图片查看器那样,鼠标滚动可以缩放,可以拖拽图片,于是就写了这个简单的图片查看器. 前台代码: <Window x:Class=&qu ...