求素数的三大算法 —— Java 篇
- 素数又叫质数(prime number),有无限个。质数定义为在大于1的自然数中,除了1和它本身以外不再有其他因数。
算法一 :暴力遍历
思路:
- 素数我们运用两个循环的遍历的方式,首先根据素数的定义:质数定义为在大于1的自然数中,除了1和它本身以外不再被其数整除了。因为任何数都可以被 1 整除的,所以我们的第一个for( )循环语句的初始值从 2 开始进行,到你所需的范围。
- 我们的第二个for()循环语句,的判断条件是:从上个for()循环语句的 变化值开始,但是注意不要等于 因为不要除以它本身,其中我们 if() 语句判断是否有,被除了它本身以外的数整除,同时我们定义一个变量,用于标记该数是否为被整除,逆向思考,被整除了,就不是素数,否则为素数
- 最后我们通过我们的标记,判断素数需要打印数值,
- 查找的时间复杂度为 O(n)
代码:
public class Study13 {
/* 求素数的三大算法*/
// 算法 —— : 暴力遍历
public static void main (String[] args) {
Scanner scanner = new Scanner (System.in);
if(scanner.hasNextInt()) {
int n = scanner.nextInt();
int count = 0;
for (int i = 2; i <= n; i++) {
int sign = 1; // 用于标记是否是素数
for (int j = 2; j < i; j++) {
if (i % j == 0 ) {
sign = 0;
break;
}
}
if (sign == 1) {
System.out.print(i+"\t");
count++;
if(count % 5 == 0) {
System.out.println();
} // 2,3,5,7,11,13,17,19,23,29,31,37,41,43,53,59,61,67,71,73,79,83,89,97。
}
}
}
scanner.close(); // 关闭 IO 流
}
}
结果:
算法二:折半范围遍历
思路:
- 在第一个算法的基础上,改进之法
- 请看下图所示:
- 我们仔细观察上述的图解:发现其规律没有:
- 所有的数的可以被整除的最大的,除数都是 {其数值的 / 2 的 }如:18 的可以被最大的整除的是 (18/2 =) 9 , 12 可以被最大的整除的数是 (12 /2 =) 6 ;
- 所以我们只要判断其数值的 1/2的范围内 是否有被整除 (除 1 外) 就可以了,会被整除就不是素数,不会整除就是素数
- 逆向思考,这样我们就缩小了遍历范围,在第一种算法的基础上减半,
- 查找的时间复杂度为 O(n/2)
代码:
import java.util.Scanner;
import java.util.concurrent.ForkJoinPool;
public class Study13 {
/* 求素数的三大算法*/
// 算法 二: 折半遍历
public static void main (String[] args) {
Scanner scanner = new Scanner (System.in);
int count = 0;
if(scanner.hasNextInt()) {
int n = scanner.nextInt();
for (int i = 2; i <= n; i++) {
int sign = 1;
for (int j = 2; j <= i/2 ; j++) {
if(i % j == 0) {
sign = 0;
break;
}
}
if(sign == 1) {
System.out.print(i+"\t");
count++;
if(count % 5 == 0) {
System.out.println();
}
}
}
}
scanner.close(); // 关闭 IO流
}
}
结果:
算法三:根号范围遍历
思路:
-
同样是在第一个算法的基础上改进的
-
请看下面的图示:
-
-
仔细观察图示,有没有发现,我们求他们的平方根看看,结果:
-
发现在它们的平方范围内,都会有可以被整除的数值,(当然我们这里不是素数的数值)逆向思考:不是的不就是素数了吗?
-
这样我们再次缩小了遍历的范围了,在第一种算法的基础上减了,一个根号范围;
-
查找的时间复杂度为 O(根号n )
代码:
import java.util.Scanner;
// 算法三.根号范围遍历
public class Study13 {
public static void main (String[] args) {
Scanner scanner = new Scanner (System.in);
int count = 0;
if(scanner.hasNextInt()) {
int n = scanner.nextInt();
for (int i = 2; i <= n; i++) {
int sign = 1;
for (int j = 2; j <= (int)Math.sqrt(i) ; j++) {
if(i % j == 0) {
sign = 0;
break;
}
}
if(sign == 1) {
System.out.print(i+"\t");
count++;
if(count % 5 == 0) {
System.out.println("");
}
}
}
}
scanner.close(); // 关闭IO 流
}
}
结果:
总结:
- 总的来说通过找寻到某种规律:从而缩小查找的范围:如第一种算法:中我们查找的范围是 n , 第二种算法中:我们查找的范围是 n / 2 ,第三中算法:我们的查找范围是 根号 n
- 从而达到实现提高程序的效率
最后:
每博一文案
你看,我早知道有很多的道理,我们并不是不知道,可人生的某些事总要自己经历过,才能真正把生活的意义都明了,没有谁可以复制谁的人生硕果,也没有谁可以规避谁的人生坎坷一句我早知道于事无补于心,有格。试着去拥抱那些满身伤口的人吧,我有陪伴和安慰,才能让那些伤口在开出花朵。
———— 一禅心灵庙语
限于自身水平有限,其中存在的错误希望大家给予指教,谢谢大家,韩信点兵——多多益善!,后会有期,江湖再见!