C语言PAT刷题 - 1007 素数对猜想

作者的话:若有朋友复制代码去PAT试着运行遇到问题的:
1.可能是格式问题,可以先把从本站复制的代码粘贴到记事本,再把记事本里的代码复制,然后粘贴到PAT的代码区,提交本题回答,应该就可以了;
2.可能是注释原因,PAT有时候检测到注释会编译错误,所以可以先把注释删了,再进行提交回答。
3.可能是作者当初根据题目写出来的代码仍存在一些疏漏,而恰好当时的测试机制没那么完善,没检测出问题。后面测试机制有所更新,故出现问题,若有相关需要的可以评论区留言或私信作者,我看到的话会去再查一下疏漏之处,然后更新文章。

一、题目描述
让我们定义dn为:dn=pn+1−pn​,其中pi是第i个素数。显然有d1=1,且对于n>1有dn是偶数。“素数对猜想”认为“存在无穷多对相邻且差为2的素数”。
现给定任意正整数N(<105),请计算不超过N的满足猜想的素数对的个数。
输入格式:
输入在一行给出正整数N。
输出格式:
在一行中输出不超过N的满足猜想的素数对的个数。
输入样例:
20
输出样例:
4

二、解题思路
读题:
首先讲一下素数的概念,素数(又称质数),指在大于1的自然数中,除了1和该数自身外,无法被其他自然数整除的数(也可定义为只有1与该数本身两个正因数的数)。
我们把相邻且相减后差为2(即满足dn=2)的两个素数称为满足猜想的一个素数对。
题目希望给一个正整数N(<10^5),程序能在不超过N的范围内,计算出有几个满足猜想的素数对。
思路:
1.定义需要的变量(实际解题时是先定义认为需要用到的变量,后面遇到问题需要新定义变量时再回到上头来定义新的变量),从键盘接收一个正整数存入变量n中;
2.定义一个countPrimePair函数来实现统计范围内素数对数量的功能;
3.调用countPrimePair函数以得到范围内素数对数量,打印。

三、具体实现
0.标准C源程序框架

#include <stdio.h>
#include <math.h>  //因为后面要使用平方根计算函数,故引用此头文件
int main()
{
    return 0;
}

1.定义需要的变量(实际解题时是先定义认为需要用到的变量,后面遇到问题需要新定义变量时再回到上头来定义新的变量),从键盘接收一个正整数存入变量n中;

	int n = 0;
	scanf("%d", &n);

2.定义一个countPrimePair函数来实现统计范围内素数对数量的功能;

int countPrimePair(int n)
{
	//除了0、2以外,其它所有的偶数都不可能是素数,因为他们会被2整除
	//而0、4不是素数,自然不可能存在和2相减为2的素数,即素数对里一定没有偶数
	//所以我们可以设置循环对所有的范围内的奇数进行检测,并在循环中设置两个判断
	//判断1.当前奇数是否为素数  判断2.当前奇数的下一个奇数是否为素数(满足判断1了才进行2的判断)
	//若当前奇数和下一个奇数符合两个判断条件,则这两个数是素数对,计数变量count++
	int count = 0;
	int i = 0;
	int j = 0;
	int result = 0;
	for (i = 3; i <= (n-2); i+=2)  //对范围内的所有奇数进行检测,又因为检测的是当前奇数和下一个奇数
	{					//所以要确保当前奇数和下一个奇数都在范围内,即i+2<=n,所以循环条件设为i<=(n-2)
		result = 1;//开始默认每个奇数都是素数,若检测出不是素数,则result置0,表示并非素数
		//判断1.当前奇数是否为素数
		for (j = 3; j <= sqrt(i); j += 2)
		{
			if (i % j == 0)//若为真,则表示i不是素数;若为假,表示当前数不是i的因数,但无法证明i是素数
			{
				result = 0;
				break;//若i不是素数,结束循环
			}
		}
		if (result == 1)//若走完上述循环,result仍为1,则表示当前奇数是素数,开始进行判断2
		{
			//判断2.当前奇数的下一个奇数是否为素数
			for (j = 3; j <= sqrt(i+2); j += 2)
			{
				if ((i+2) % j == 0)//若为真,则表示(i+2)不是素数;
				{				   //若为假,表示当前数不是(i+2)的因数,但无法证明(i+2)是素数
					result = 0;
					break;//若(i+2)不是素数,结束循环
				}
			}
			if (result == 1)//如果对(i+2)的检测走完之后,result仍为1,则表示(i+2)也是素数
			{
				count++;
			}
		}
	}
	return count;
}

3.调用countPrimePair函数以得到范围内素数对数量,打印。

printf("%d",countPrimePair(n));
//将变量n传参传给函数countPrimePair(),调用函数计算出范围内满足猜想的素数对个数,结果传回主函数,对这个结果进行输出

四、全部代码

#include <stdio.h>
#include <math.h>
int countPrimePair(int n)
{
	//除了0、2以外,其它所有的偶数都不可能是素数,因为他们会被2整除
	//而0、4不是素数,自然不可能存在和2相减为2的素数,即素数对里一定没有偶数
	//所以我们可以设置循环对所有的范围内的奇数进行检测,并在循环中设置两个判断
	//判断1.当前奇数是否为素数  判断2.当前奇数的下一个奇数是否为素数(满足判断1了才进行2的判断)
	//若当前奇数和下一个奇数符合两个判断条件,则这两个数是素数对,计数变量count++
	int count = 0;
	int i = 0;
	int j = 0;
	int result = 0;
	for (i = 3; i <= (n-2); i+=2)  //对范围内的所有奇数进行检测,又因为检测的是当前奇数和下一个奇数
	{					//所以要确保当前奇数和下一个奇数都在范围内,即i+2<=n,所以循环条件设为i<=(n-2)
		result = 1;//开始默认每个奇数都是素数,若检测出不是素数,则result置0,表示并非素数
		//判断1.当前奇数是否为素数
		for (j = 3; j <= sqrt(i); j += 2)//若i=m*n,m和n如果不相等,一定是一个>sqrt(i),一个<sqrt(i)
		{//因为若m,n都<sqrt(i),则m*n<(sqrt(i))^2=i,所以m,n一定是一个大一点,一个小一点,除非m=n=sqrt(i)。因此找i除了1和自己以外的因数只需找sqrt(i)之前的数即可
			if (i % j == 0)//若为真,则表示i不是素数;若为假,表示当前数不是i的因数,但无法证明i是素数
			{
				result = 0;
				break;//若i不是素数,结束循环
			}
		}
		if (result == 1)//若走完上述循环,result仍为1,则表示当前奇数是素数,开始进行判断2
		{
			//判断2.当前奇数的下一个奇数是否为素数
			for (j = 3; j <= sqrt(i+2); j += 2)
			{
				if ((i+2) % j == 0)//若为真,则表示(i+2)不是素数;
				{				   //若为假,表示当前数不是(i+2)的因数,但无法证明(i+2)是素数
					result = 0;
					break;//若(i+2)不是素数,结束循环
				}
			}
			if (result == 1)//如果对(i+2)的检测走完之后,result仍为1,则表示(i+2)也是素数
			{
				count++;
			}
		}
	}
	return count;
}
int main()
{
	int n = 0;
	scanf("%d", &n);
	printf("%d",countPrimePair(n));//将变量n传参传给函数countPrimePair(),调用函数计算出范围内满足猜想
	return 0;                      //的素数对个数,结果传回主函数,对这个结果进行输出
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值