自然数的拆分

简介: 【问题描述】自然数的拆分:任何一个大于1的自然数N,总可以拆分成若干个自然数之和,并且有多种拆分方法。试求 n的所有拆分。例如自然数5,可以有如下一些拆分方法:5=1+1+1+1+15=1+1+1+25=1+2+25=1+45=2+3 注意,本题中N拆分出来的数x的范围是11) 25 ...

【问题描述】自然数的拆分:任何一个大于1的自然数N,
总可以拆分成若干个自然数之和,并且有多种拆分方法。试求 n的所有拆分。
例如自然数5,可以有如下一些拆分方法:
5=1+1+1+1+1
5=1+1+1+2
5=1+2+2
5=1+4
5=2+3

注意,本题中N拆分出来的数x的范围是1<=x<N。假如x可以等于N,那么本题和整数划分是一个道理。现在,这里与整数划分这个题的答案只少1,即N拆分成N本身的情况。

整数划分可以参考:

http://www.cnblogs.com/hoodlum1980/archive/2008/10/11/1308493.html

http://blog.csdn.net/sunquana/article/details/9245443

 

算法一  用回溯法来实现

针对所给问题,定义问题的解空间;如本题对5的拆分来说,1<=拆分的数<5

确定用于搜索的解空间结构;如本题对5的拆分来说,x[ ]数组来存储解,每个数组元素的取值范围都是1<=拆分的数<=5,从1开始搜索直到5 

搜索解空间,并在搜索过程中用剪枝函数避免无效搜索。

如本题对5的拆分来说,为了避免重复,x[i] >= x[j]  ( i > j ),如x[]={2,3}满足条件而x[]={3,2}就不满足条件不是可行解即无效

 1 #include<stdio.h>
 2 #include<stdlib.h>
 3 
 4 void splitN(int n,int m);// n是需要拆分的数,m是拆分的进度。
 5 int x[1024]={0},total=0 ;// total用于计数拆分的方法数,x[]用于存储解
 6 void main()
 7 {
 8     int n ;
 9     printf("please input the natural number n:");
10     scanf("%d",&n);
11     splitN(n,1);
12     printf("There are %d ways to split natural number %d. ",total,n);
13 }
14 
15 void splitN(int n,int m)
16 {//n是需要拆分的数,m是拆分的进度
17     int rest,i,j;
18     for(i=1;i<=n;i++)
19     {//从1开始尝试拆分
20         if(i>=x[m-1])
21         {//拆分的数大于或等于前一个从而保证不重复
22             x[m]=i ;// 将这个数计入结果中
23             rest=n-i ;// 剩下的数是n-i,如果已经没有剩下的了,并且进度(总的拆分个数)大于1,说明已经得到一个结果了
24             if(rest==0&&m>1)
25             {
26                 total++;
27                 printf("%d\t",total);
28                 for(j=1;j<m;j++)
29                 {
30                     printf("%d+",x[j]);
31                 }
32                 printf("%d ",x[m]);
33                 printf("\n");
34             }
35             else
36             {
37                 splitN(rest,m+1);// 否则将剩下的数进行进度为m+1拆分
38             }
39             x[m]=0;// 取消本次结果,进行下一次拆分。环境恢复,即回溯
40         }
41     }
42 }
View Code

 

 

算法二  用递归来实现

用不完全归纳法
n =2 可拆分成 2 =1 +1
n =3 可拆分成 3 =1 +2 =1 +1 +1
n =4 可拆分成 4 =1 +3 =1 +1 +2 =1 +1 +1 +1 =2 +2
……
n =7 可拆分成 7=1 +6
=1 +1 +5
=1 +1 +1 +4
=1 +1 +1 +1 +3
=1 +1 +1 +1 +1 +2
=1 +1 +1 +1 +1
=1 +1 +1 +2 +2
=1 +1 +2 +3
=1 +2 +4
=1 +2 +2 +2
=1 +3 +3

=2 +5
=2 +2 +3

=3 +4

用数组 a 存储完成 n 的一种拆分。从上面不完全归纳法的分析 n =7 时,
按 a[1]分类,有a[1]=1,a[1]= 2,…,a[1]= n/2,共 n/2 大类拆分。
在每一类拆分时,a[1]= i ,a[2]= n - i ,从 k=2,从 a[k]开始继续拆分,
a[k]能否再拆分取决于 a[k]/2 是否大于等于 a[k-1]。
递归过程的参数 t 指向要拆分的数 a[k]

 1 #include <stdio.h>
 2 #include <stdlib.h>
 3 
 4 int a[100]={0};
 5 void Split(int t)
 6 {
 7     int i,j,L;
 8     for(i = 1; i <t; i++)
 9     {
10         printf("%d+",a[i]);
11     }
12     printf("%d\n",a[i]);
13 
14     j = t;
15     L = a[j];
16     for(i = a[j-1]; i <= L/2; i++)
17     {
18         a[j] = i;
19         a[j+1] = L - i;
20         Split(j+1);
21     }
22 }
23 
24 void SplitNum(int n)
25 {
26     int i;
27     for(i = 1; i <= n/2; i++)
28     {
29         a[1] = i;
30         a[2] = n - i;
31         Split(2);
32     }
33 }
34 int main()
35 {
36     int n;
37     scanf("%d",&n);
38     SplitNum(n);
39     return 0;
40 }
View Code

 

参考:http://wenku.baidu.com/link?url=H7tDqvEmnds9SN4FfuwMw8M6AfAMUl44-vCR83Z4LKv9UN-HAU159GJtFR5M48t11XBJFMwP3i4qPk6u2WHEORZDeYhraBQt63zvaDUNSAi

 

相关文章
|
Java Maven
开源一套原创文本处理工具:Java+Bat脚本实现自动批量处理对账单工具
这款工具是笔者在2018年初开发完成的,时隔两载,偶然想起这款小工具,于是,决定将其开源,若有人需要做类似Java批处理实现整理文档的工具,可参考该工具逻辑思路来实现。
210 0
|
Linux 关系型数据库 Oracle
|
监控 安全 数据安全/隐私保护
系统监控软件有哪些
【10月更文挑战第17天】
659 10
|
监控 Java 微服务
Spring Boot微服务部署与监控的实战指南
【7月更文挑战第19天】Spring Boot微服务的部署与监控是保障应用稳定运行和高效维护的重要环节。通过容器化部署和云平台支持,可以实现微服务的快速部署和弹性伸缩。而利用Actuator、Prometheus、Grafana等监控工具,可以实时获取应用的运行状态和性能指标,及时发现并解决问题。在实际操作中,还需根据应用的具体需求和场景,选择合适的部署和监控方案,以达到最佳效果。
|
存储 算法 索引
【头歌·计组·自己动手画CPU】三、存储系统设计(HUST)(理论版) 【计算机硬件系统设计】
【头歌·计组·自己动手画CPU】三、存储系统设计(HUST)(理论版) 【计算机硬件系统设计】
2176 1
|
自然语言处理 监控 搜索推荐
《百炼成金-大金融模型新篇章》––12.应用场景与技术架构选型(1)
百炼必定成金,新质生产力会催生新质劳动力,谨以此文抛砖引玉,希望与业内的各位朋友一同探讨如何积极拥抱并运用大模型技术,以应对和驾驭不断变化的市场环境,实现科技金融持续稳定的提质增效和创新发展,携手开启金融大模型未来新篇章。
578 1
|
存储
【头歌·计组·自己动手画CPU】四、控制器设计(理论版) 【计算机硬件系统设计】
【头歌·计组·自己动手画CPU】四、控制器设计(理论版) 【计算机硬件系统设计】
1214 0
|
存储 JSON Java
2024要做财务报表 有哪些报表工具可以推荐
财务报表是企业DNA,需多角度分析。润乾报表擅长复杂报表,提供全面功能,类Excel界面,支持多样数据源,高效替代传统数据准备方式,集成简单,价格亲民。帆软报表也是Java Web报表工具,兼容多种数据库,内置BI,但集成可能复杂,性能在大数据量时受影响,视觉效果出色。两者都能满足财务报表需求,选择取决于具体业务场景和预算。
342 0