题目背景

1742年6月7日哥德巴赫写信给当时的大数学家欧拉,正式提出了以下的猜想:任何一个大于9的奇数都可以表示成3个质数之和。质数是指除了1和本身之外没有其他约数的数,如2和11都是质数,而6不是质数,因为6除了约数1和6之外还有约数2和3。需要特别说明的是1不是质数。

这就是哥德巴赫猜想。欧拉在回信中说,他相信这个猜想是正确的,但他不能证明。

从此,这道数学难题引起了几乎所有数学家的注意。哥德巴赫猜想由此成为数学皇冠上一颗可望不可及的“明珠”。

题目描述

现在请你编一个程序验证哥德巴赫猜想。

先给出一个奇数n,要求输出3个质数,这3个质数之和等于输入的奇数。

输入格式

仅有一行,包含一个正奇数n,其中9<n<20000

输出格式

仅有一行,输出3个质数,这3个质数之和等于输入的奇数。相邻两个质数之间用一个空格隔开,最后一个质数后面没有空格。如果表示方法不唯一,请输出第一个质数最小的方案,如果第一个质数最小的方案不唯一,请输出第一个质数最小的同时,第二个质数最小的方案。

输入输出样例

输入 #1

2009

输出 #1

3 3 2003


我的分析

此题就是暴力递归搜索,本身没什么难度,但是如果用常规的枚举2到sqrt(n)之间的数来对n进行素性检测的方法,最后一个样例就过不了!重要的事情说三遍,过不了!过不了!过不了!(超凶╭(╯^╰)╮)

于是,我又去网上找了一个改进后的素性测试方法(虽然搞不懂什么原理O(∩_∩)O),然后成功地通过了所有样例

  1. #include<iostream>
  2. #include<cmath>
  3. #include<cstring>
  4. using namespace std;
  5. //素性检测函数
  6. bool isprime(int n){
  7. int p[8]={4,2,4,2,4,6,2,6};
  8. int i=7,j,q;
  9. if(n==1)return 0;
  10. if(n==2||n==5||n==3)return 1;
  11. if(n%2==0||n%3==0||n%5==0)return 0;
  12. q=(int)sqrt(n);
  13. for(;i<=q;){
  14. for(j=0;j<8;j++){
  15. if(n%i==0)return 0;
  16. i+=p[j];
  17. }
  18. if(n%i==0)return 0;
  19. }
  20. return 1;
  21. }
  22. //tmp数组用来暂时缓存可能成为答案的3个数,res数组用来储存最终答案的3个数
  23. int res[4],tmp[4];
  24. void dfs(int n,int count){
  25. //如果已经积累到2个素数了,那么本轮递归传入的n就是剩下的一个数了
  26. //若n为素数,则产生一组可能答案,以下步骤尝试更新最终结果
  27. if(count==2&&isprime(n)){
  28. tmp[++count]=n;
  29. if(tmp[1]<res[1]){
  30. res[1]=tmp[1];
  31. res[2]=tmp[2];
  32. res[3]=tmp[3];
  33. }
  34. else if(tmp[1]==res[1]){
  35. if(tmp[2]<res[2]){
  36. res[1]=tmp[1];
  37. res[2]=tmp[2];
  38. res[3]=tmp[3];
  39. }
  40. }
  41. return;
  42. }
  43. //若已经积累2个素数,而剩下的一个数非素数,则返回
  44. else if(count==2) return;
  45. for(int i=2;i<n;++i){
  46. if(isprime(i)){
  47. tmp[++count]=i;
  48. dfs(n-i,count);
  49. count--; //注意递归返回后计数器也一定要跟着复原
  50. }
  51. }
  52. }
  53. int main(){
  54. //初始化两个数组,注意初始时res比所给的数大,而tmp一定要比res大
  55. memset(res,20001,sizeof(int)*4);
  56. memset(tmp,20002,sizeof(int)*4);
  57. int n;
  58. cin>>n;
  59. dfs(n,0);
  60. cout<<res[1]<<" "<<res[2]<<" "<<res[3];
  61. return 0;
  62. }

洛谷P1579.验证哥德巴赫猜想(DFS+素性测试)的更多相关文章

  1. PTA实验4-2-3 验证“哥德巴赫猜想” (20分)

    实验4-2-3 验证"哥德巴赫猜想" (20分) 数学领域著名的"哥德巴赫猜想"的大致意思是:任何一个大于2的偶数总能表示为两个素数之和.比如:24=5+19, ...

  2. 洛谷——P1579 哥德巴赫猜想(升级版)

    P1579 哥德巴赫猜想(升级版) 题目背景 1742年6月7日哥德巴赫写信给当时的大数学家欧拉,正式提出了以下的猜想:任何一个大于9的奇数都可以表示成3个质数之和.质数是指除了1和本身之外没有其他约 ...

  3. 洛谷 P1579 哥德巴赫猜想(升级版)【筛素数/技巧性枚举/易错】

    [链接]:https://www.luogu.org/problemnew/show/P1579 题目背景 1742年6月7日哥德巴赫写信给当时的大数学家欧拉,正式提出了以下的猜想:任何一个大于9的奇 ...

  4. 洛谷P1579 哥德巴赫猜想(升级版)【水题+素数】

    1742年6月7日哥德巴赫写信给当时的大数学家欧拉,正式提出了以下的猜想:任何一个大于9的奇数都可以表示成3个质数之和.质数是指除了1和本身之外没有其他约数的数,如2和11都是质数,而6不是质数,因为 ...

  5. Java实现 洛谷 P1579 哥德巴赫猜想(升级版)

    题目背景 1742年6月7日哥德巴赫写信给当时的大数学家欧拉,正式提出了以下的猜想:任何一个大于9的奇数都可以表示成3个质数之和.质数是指除了1和本身之外没有其他约数的数,如2和11都是质数,而6不是 ...

  6. 洛谷P1219 八皇后【dfs】

    题目描述 检查一个如下的6 x 6的跳棋棋盘,有六个棋子被放置在棋盘上,使得每行.每列有且只有一个,每条对角线(包括两条主对角线的所有平行线)上至多有一个棋子. 上面的布局可以用序列2 4 6 1 3 ...

  7. 洛谷1378 油滴扩展 dfs进行回溯搜索

    题目链接:https://www.luogu.com.cn/problem/P1378 题目中给出矩形的长宽和一些点,可以在每个点放油滴,油滴会扩展,直到触碰到矩形的周边或者其他油滴的边缘,求出剩余面 ...

  8. c语言验证哥德巴赫猜想(从4开始 一个偶数由两个质数之和)

    #include <stdio.h> #include <stdlib.h> #include <math.h> int isit(int num) { int i ...

  9. 洛谷 P1579 哥德巴赫猜想(升级版)

    嗯... 这或许也算一道数论题吧... 题目链接:https://www.luogu.org/problemnew/show/P1579 这道题的说明好像只会扰乱人的思路....然后就是这道题的细节比 ...

随机推荐

  1. SDS——重用StringBuilder

    package example.java; /** * @author 杜科 * @description 简单动态字符串,非线程安全.采取类似buffer的设计,使其成为一个可以方便重用的Strin ...

  2. 01 shell编程之变量定义

    一.SHELL介绍 ㈠ 什么是shell脚本? 简单来说就是将需要执行的命令保存到文本中,按照顺序执行.它是解释型的,意味着不需要编译. 若干命令 + 脚本的基本格式 + 脚本特定语法 + 思想= s ...

  3. vue学习(十三) 删除对象数组中的某个元素

    //html <div id="app"> //v-for循环就不写了 每一条数据最后都有一个删除的超链 .prevent阻止默认的跳转行为 只执行点击事件 <a ...

  4. [难题题解] [BZOJ1875] [SDOI2009] HH去散步

    题目H有个一成不变的习惯,喜欢饭后百步走.所谓百步走,就是散步,就是在一定的时间 内,走过一定的距离. 但是同时HH又是个喜欢变化的人,所以他不会立刻沿着刚刚走来的路走回. 又因为HH是个喜欢变化的人 ...

  5. 了不起的Node.js 5/16

    Chapter 1 安装 1.Node.js的设计理念之一,只维护较少量的依赖,这使得安装node.js变得非常简单. 2.执行文件console.log没问题,但是node执行http静态服务器的时 ...

  6. Django创建项目时应该要做的几件事

    终于可以在假期开始学习 Django 啦 !

  7. PHP is_link() 函数

    定义和用法 The is_link() 函数检查指定的文件是否是一个连接. 如果文件是一个连接,该函数返回 TRUE. 语法 is_link(file) 参数 描述 file 必需.规定要检查的文件. ...

  8. PHP preg_replace() 函数

    preg_replace 函数执行一个正则表达式的搜索和替换.高佣联盟 www.cgewang.com 语法 mixed preg_replace ( mixed $pattern , mixed $ ...

  9. PHP mysqli_refresh() 函数

    定义和用法 mysqli_refresh() 函数刷新表或缓存,或者重置复制服务器信息.高佣联盟 www.cgewang.com 语法 mysqli_refresh(connection,option ...

  10. 下载excel模板,导入数据时需要用到

    页面代码: <form id="form1" enctype="multipart/form-data"> <div style=" ...