`

[转]打印质数的各种算法

阅读更多

打印质数的算法应该是学习计算机编程的一个经典的问题,在这里想给大家展示一些方法,相信这些方法会对你的编程有一定的启发作用。请你注意几点,

  • 实际应用和教学应用有很大的差别。
  • 最后的那个使用编译时而不是运行时的方法大家可以重点看看。

教科书的示例

首先,先给一个教科书的示例。下面这个示例应该是教科书(至少是我上大学时的教科学)中算法复杂度最好的例子了。其想法很简单,先写一个判断是否是 质数的函数isPrime(),然后从1到n分别调用isPrime()函数来检查。检查是否是质数的算法是核心,其简单的使用从2到n的开根的数作为除 数。这样的算法复杂度几乎是O(n*log(n)),看上去不错,但其实很不经济。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
#include <iostream>
using namespace std;
 
bool isPrime( int nr)
{
     for ( int d = 2; (d * d) < (nr + 1); ++d){
         if (!(nr % d)){
             return false ;
         }
      }
     return true ;
}
 
int main ( int argc, char * const argv[])
{
     for ( int i = 0; i < 50; ++i){
         if (isPrime(i)){
             cout << i << endl;
         }
     }
}

较好的算法

我们知道,我们的算法如果写成线性算法,也就是O(n),已经算是不错了,但是最好的是O(Log(n))的算法,这是一个级数级的算法,著名的二分取中(Binary Search)正是O(Log(n))的算法。通常来说,O(Log(n))的算法都是以排除法做为手段的 。所以,找质数的算法完全可以采用排除法的方式。如下所示,这种算法的复杂度是O (n(log(logn)))。

示例:打印30以内的质数

一、初始化如下列表。

 2  3  4  5  6  7  8  9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30

二、把第一个数(2)取出来,去掉所有可以被2整除的数。

 2  3     5     7     9    11    13    15    17    19    21    23    25    27    29

三、取第二个数(3),去掉所有可以被 3整除的数。

 2  3     5     7          11    13          17    19          23    25          29

四、取第三个数(5),因为4已经被去除了,再去掉所有可以被5整除的数。

 2  3     5     7          11    13          17    19          23                29

接下来的数是7,但是7的平方是49,其大于了30,所以我们可以停止计算了。剩下的数就是所有的质数了。

实际应用的算法

实际应用中,我们通常不会使用上述的两种算法,因为那是理论学院派的算法。实际中的算法是,我把质数事先就计算好,放在一个文件中,然后在程序启动 时(注意是在启动时读这个文件,而不是运行时每调用一次就读一次文件),读取这个文件,然后打印出来就可以了。如果需要查找的化,二分查找或是hash表 查找将会获得巨大的性能提升。当然,这样的方法对于空间来说比前面两个都要消耗得大,但是你可以有O(log(n))或是O(1)的时间复杂度。

所以,我想在这里提醒大家——实际和理论的的方法很不一样的 ,千万不要读书读成书呆子。在游戏编程的世界里,大量的数据都不是运行计算的,而都是写在文件中的。比如,一个火焰效果,一个人物跑动的动作,都是事先写在文件中的。

使用编译时而不是运行时

下面这个例子(本例参考于这里 )你需要注意了,这是一个高级用法,使用模式来在编译时计算质数,而不是运行时。这种技术使用了C++编译器对模板的特化时的处理来生成自己相要的结果。这种方法在技术上是相当Cool的,但并不一定实用,这里只是想像大家展示这种用法。这是C++的最骨灰级的用法了。

请看下面的两个模板类,第一个模板以递归的方式检查是否是质数,第二个方法是递归的退出条件(当N=1时),对于模板的重载,请参看相关的C++书籍。

1
2
3
4
5
6
7
8
9
10
11
12
13
template < int N, int D = N - 1>
struct isPrime {
     enum {
         result = (N % D) && isPrime<N, D-1>::result
     };
};
 
template < int N>
struct isPrime<N, 1> {
     enum {
         result = true
     };
};

于是,通过这个模板,我们可以使用下面的代码来检查是否是质数:

1
2
if (isPrime<3>::result)
     cout << "Guess what: 3 is a prime!" ;

下一步,我们需要打出一个区间内的质数,所以,我们需要继续设计我们的print模板。

1
2
3
4
5
6
7
8
9
10
11
template < int N, bool ISPRIME>
struct printIfPrime {
     static inline void print() {}
};
 
template < int N>
struct printIfPrime<N, true > {
     static inline void print() {
         std::cout << N << endl;
     }
};

从上面的代码中,我们可以看到,我们的第一个实际是什么也没做,而第二个有输出,注意第二个的模板参数中有一个true,其意味着那个质数的判断。于是我们就可以给出下面的代码来尝试着打印出一段区间内的质数:(请不要编译!! 因为那会让编译器进入无限循环中,原因是printPrimes会不停地调用自己永不停止)

1
2
3
4
5
6
7
8
template < int N, int MAX>
struct printPrimes {
     static inline void print()
     {
         printIfPrime<N, isPrime<N>::result>::print();
         printPrimes<N + 1, MAX>::print();
     }
};

为了避免这个问题,你需要再加一个模板类,如下所示。这样当N变成MAX的时候,递归就结束了。

1
2
3
4
5
6
template < int N>
struct printPrimes<N, N> {
     static inline void print() {
         printIfPrime<N, isPrime<N>::result>::print();
     }
};

最后,让我们来看看最终的调用:

1
2
3
4
5
int main ( int argc, char * const argv[])
{
     printPrimes<2, 40>::print();
     return 0;
}

这个方法很NB,但是有两个问题:

  • 比较耗编译时间。
  • 不能在运行时输入MAX的值。

不过,相信这种玩法会启动你很多的编程思路。

当然,还有以前说过的那个——《检查素数的正则表达式

 

原文地址:http://coolshell.cn/articles/3738.html

分享到:
评论

相关推荐

    概率算法求素数(c语言)

    ### 概率算法求素数(C语言) #### 实验目的与背景 本实验旨在通过C语言实现一种基于概率的方法来查找一定范围内的素数。概率算法是一种在计算机科学和数学领域广泛应用的技术,特别是在处理复杂问题时,它可以...

    Java中素数算法非常实用

    ### Java中的素数算法 #### 一、引言 在计算机科学领域,特别是算法与数据结构的研究中,素数检测是一项基本且重要的任务。本文将详细介绍一个简单的Java程序,该程序能够有效地找出并打印出前500个素数。 #### ...

    java实现打印素数/质数程序

    在编程领域,素数或质数是指大于1且仅能被1和其本身整除的自然数。在Java中实现打印素数的程序是一项基础但重要的任务,这有助于理解和掌握循环、条件判断以及数学逻辑在编程中的应用。下面将详细解释如何通过Java...

    素数的最好算法.

    给定的代码片段是用C语言实现的一种素数筛选算法,主要目标是在101到200之间找出所有的素数,并打印出来。这段代码的关键在于其算法设计,即如何高效地判断一个数是否为素数。 ### 算法分析 #### 算法核心:优化的...

    对质数进行枚举,这是算法类问题的解答

    - `r`: 控制是否打印质数的标志位。 - **内部逻辑**: - 遍历从1到`i`的每一个整数`j`。 - 若`i`能被`j`整除,则`k`加1。 - 如果`k`恰好等于2且`i`等于`j`,则说明`i`是一个质数: - 如果`r`为1,则输出该质数...

    JAVA经典算法大全(都会就算牛逼的了)

    这个算法解决了打印出所有的 "水仙花数" 的问题,即一个三位数,其各位数字立方和等于该数本身。算法使用for循环来控制100-999个数,每个数分解出个位、十位、百位,并判断是否为水仙花数。该算法展示了for循环语句...

    JAVA算法编程题目及答案.doc

    JAVA算法编程题目及答案 本资源提供了50道JAVA算法编程题目及答案,涵盖了算法设计、数据结构、程序设计等多个方面的知识点。以下是对标题、描述、标签和部分内容的详细解释: 标题:JAVA算法编程题目及答案 本...

    数组求和,素数,排序算法

    6. **素数判断算法**: - 在`sushu()`函数中,对于每个数组元素,从2开始到元素的一半(不包括自身)进行遍历,如果找到能整除该元素的数,则该元素不是素数,跳出循环。如果循环结束都没有找到能整除的数,说明是...

    C经典算法之Eratosthenes筛选求质数

    最后,通过输出循环打印出了所有小于`N`的质数。 #### 性能分析 Eratosthenes筛法的时间复杂度大致为O(n log log n),这使得它成为一种高效的求解质数的方法。此外,该算法的空间复杂度为O(n),因为需要一个大小为...

    vb算法,求素数,用程序实现

    在VB(Visual Basic)编程语言中,实现求素数的算法是一项基础且重要的任务。素数是指大于1的自然数,除了1和它自身以外,不能被其他自然数整除的数。例如,2、3、5、7、11等都是素数。下面我们将详细探讨如何使用VB...

    java中比较常见的经典算法

    一、九九乘法表打印算法 打印九九乘法口诀表是 Java 中经典的算法之一。通过使用 for 循环,可以实现九九乘法表的打印。下面是实现代码: ```java public void nineNineMulitTable(){ for (int i = 1,j = 1; j ; i...

    c语言100个经典算法 c语言经典算法100例

    程序分析:对 n 进行分解质因数,应先找到一个最小的质数 k,然后按下述步骤完成:(1)如果这个质数恰等于 n,则说明分解质因数的过程已经结束,打印出即可。(2)如果 n&lt;&gt;k,但 n 能被 k 整除,则应打印出 k 的值,...

    JAVA经典算法面试39题及答案

    题目:打印出所有的“水仙花数”,所谓“水仙花数”是指一个三位数,其各位数字立方和等于该数本身。 答案:使用 for 循环控制 100-999 个数,每个数分解出个位、十位、百位,然后判断是否满足“水仙花数”的条件。...

    JAVA经典算法40面试题

    该问题的解决方案使用了数学方法,即对n进行分解质因数,应先找到一个最小的质数k,然后按下述步骤完成:如果这个质数恰等于n,则说明分解质因数的过程已经结束,打印出即可。如果n &lt;&gt; k,但n能被k整除,则应打印出k...

    打印出100以内的质数Java

    在这里,我们将编写打印质数的逻辑。 - **循环遍历**:使用`for`循环从2开始遍历到100。 - **质数检查**:对每个数字i,用另一个`for`循环从2到Math.sqrt(i)检查是否有因数。 - **输出**:如果找到的数字i在遍历...

    Java求质数的几种常用算法分析

    "Java求质数的几种常用算法分析" 本文主要介绍了Java中求质数的几种常用算法,并结合实例形式分析了三种比较常见的求质数算法原理及相关实现技巧。 一、根据质数的定义求质数 质数的定义是:只能被1或者自身整除...

    单片机C语言常用算法

    单片机C语言常用算法是指在单片机系统中使用C语言实现的算法,包括计数、求和、求阶乘、求最大公约数、最小公倍数、判断素数、验证哥德巴赫猜想等。这些算法都是解决实际问题的基本思想方法和步骤。 一、计数、求和...

    100个经典算法.docx

    本篇文档提供了100个使用C语言实现的经典算法例题,涵盖了各种类型的算法,包括数学问题、数论问题和逻辑问题等。以下是对这些算法的详细说明: 1. **兔子问题**(Fibonacci数列):这是基于斐波那契数列的一个问题...

    Java经典算法40例

    本文通过解析给定文件中的部分文字内容,旨在详细介绍Java经典算法40例中的三个具体实例,这些实例涉及递归算法、素数判定以及水仙花数的识别。以下是针对每个程序分析及知识点的详细介绍: 程序1:斐波那契数列...

    java笔试常见的算法题

    全排序、二分查找、冒泡排序、阶乘、最大公约数、最小公倍数、打印九九乘法表、判断素数、快速排序的递归实现和非递归实现、随机数、字符串操作、50人围成一圈,数到3和3的倍数的人出局,最后剩下的人是谁。...

Global site tag (gtag.js) - Google Analytics