求素数的三大算法 —— C 语言 篇

求素数的三大算法 —— C 语言 篇


  • 素数又叫质数(prime number),有无限个。质数定义为在大于1的自然数中,除了1和它本身以外不再有其他因数。

算法一 :暴力遍历

思路:

  1. 素数我们运用两个循环的遍历的方式,首先根据素数的定义:质数定义为在大于1的自然数中,除了1和它本身以外不再被其数整除了。因为任何数都可以被 1 整除的,所以我们的第一个for( )循环语句的初始值从 2 开始进行,到你所需的范围。
  2. 我们的第二个for()循环语句,的判断条件是:从上个for()循环语句的 变化值开始,但是注意不要等于 因为不要除以它本身,其中我们 if() 语句判断是否有,被除了它本身以外的数整除,同时我们定义一个变量,用于标记该数是否为被整除,逆向思考,被整除了,就不是素数,否则为素数
  3. 最后我们通过我们的标记,判断素数需要打印数值,
  4. 查找的时间复杂度为 O(n)

代码:


// 暴力遍历法
int main()
{
	int n = 0;
	int count = 0;
	scanf("%d", &n);
	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)
		{
			printf("%d ", i);
			count++;
			if (count % 5 == 0)
			{
				printf("\n");
			}

		}
		

	}

	return 0;
}


算法二:折半范围遍历

思路:

  • 在第一个算法的基础上,改进之法
  • 请看下图所示:

在这里插入图片描述

  • 我们仔细观察上述的图解:发现其规律没有:
    • 所有的数的可以被整除的最大的,除数都是 {其数值的 / 2 的 }如:18 的可以被最大的整除的是 (18/2 =) 9 , 12 可以被最大的整除的数是 (12 /2 =) 6 ;
    • 所以我们只要判断其数值的 1/2的范围内 是否有被整除 (除 1 外) 就可以了,会被整除就不是素数,不会整除就是素数
  • 逆向思考,这样我们就缩小了遍历范围,在第一种算法的基础上减半,
  • 查找的时间复杂度为 O(n/2)

代码:

#include<
// 折半范围遍历
int main()
{
	int n = 0;
	scanf("%d", &n);
	int count = 0;
	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)
		{
			printf("%d\t", i);
			count++;
			if (count % 5 == 0)
			{
				printf("\n");
			}
		}
	}

	return 0;
}

算法三:根号范围遍历

思路:

  • 同样是在第一个算法的基础上改进的

  • 请看下面的图示:

    • 在这里插入图片描述

  • 仔细观察图示,有没有发现,我们求他们的平方根看看,结果:

  • 发现在它们的平方范围内,都会有可以被整除的数值,(当然我们这里不是素数的数值)逆向思考:不是的不就是素数了吗?

  • 这样我们再次缩小了遍历的范围了,在第一种算法的基础上减了,一个根号范围;

  • 查找的时间复杂度为 O(根号n )


代码:

#include<stdio.h>
#include<math.h>
// 根号范围遍历
int main()
{
	int n = 0;
	int count = 0;
	scanf("%d", &n);
	for (int i = 2; i <= n; i++)
	{
		int sign = 1;
		for (int j = 2; j <= (int)sqrt(i); j++) // sqrt 求平方根的函数,返回                                                             // 值是 double ,所以这里强转一下
		{
			if (i % j == 0)
			{
				sign = 0;
				break;
			}
		}
		if (sign == 1)
		{
			printf("%d\t", i);
			count++;
			if (count % 5 == 0)
			{
				printf("\n");
			}
		}
	}


	return 0;
}

总结:

  • 总的来说通过找寻到某种规律:从而缩小查找的范围:如第一种算法:中我们查找的范围是 n , 第二种算法中:我们查找的范围是 n / 2 ,第三中算法:我们的查找范围是 根号 n
  • 从而达到实现提高程序的效率

最后:

每博一文案

与其把时间花在争论上,不如多花些点时间提高自己?

让那些不懂你的人望尘莫及,有些话不说,有些事不争人生就是个不断成熟的过程。

等你真正得到境界的提升,你就会发现,与你志同道合的人越来越多,不说不争,也能找到中窝

​ ———— 一禅心灵庙语

限于自身水平有限,其中存在的错误希望大家给予指教,谢谢大家,韩信点兵——多多益善!,后会有期,江湖再见!

  • 2
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值