分治算法——半数集问题

阅读: 评论:0

分治算法——半数集问题

分治算法——半数集问题

半数集问题

什么是半数集?
给定一个自然数n,在n的左边加上一个自然数,但该自然数不能超过最近添加的数的一半。
举个例子:若n=6,求6的半数集。
{6,16,26,126,36,136}
6半数集肯定包含他本身;
1不大于的一半 所以16;
2不大于6的一半,26;1又不大于2,所以126;
3不大于6的一半,36;1又不大于3,所以136;

解题思路


由图可知,12的半数集依赖于1至6的半数集。
1(本身)+1的半数集个数+2的+…+6的半数集个数=12的半数集个数。
而每个子半数集又依赖于它们的子半数集,则可利用递归调用实现算法。
得出一个公式:

具体实现

int comp(int n){		int ans=1;		//定义变量 初始为1,为该数自身;if(n>1){			//若求1的半数集 则就是1,若大于1 则执行;for(int i=1;i<=n/2;i++)			//循环从1 加到 n/2 半数集ans+=comp(i);}return ans;						//返回半数集个数
}

以上的代码实现有一些重复的计算,例如算12的半数集时已经算过了1-6的半数集,而算6的半数集时算了1-3的半数集,1-3的半数集重复了多次。

//记忆式搜索
int a[100];		//定义一个数组,来存储对应半数集个数
int comp(int n){		int ans=1;		//定义变量 初始为1,为该数自身;if(a[n]>0)return a[n];	//如果a[n]>0,证明n的半数集已经被算过,保存到了数组a[n]里,直接返回;for(int i=1;i<=n/2;i++)			//循环从1 加到 n/2 半数集ans+=comp(i);a[n]=ans;				//如果n的半数集没被算过,现在算过了,保存到a[n]中,返回ans或者a[n]都行。return ans;						//返回半数集个数
}

本文发布于:2024-01-29 14:31:58,感谢您对本站的认可!

本文链接:https://www.4u4v.net/it/170650992115946.html

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系,我们将在24小时内删除。

下一篇:实现题2
标签:算法
留言与评论(共有 0 条评论)
   
验证码:

Copyright ©2019-2022 Comsenz Inc.Powered by ©

网站地图1 网站地图2 网站地图3 网站地图4 网站地图5 网站地图6 网站地图7 网站地图8 网站地图9 网站地图10 网站地图11 网站地图12 网站地图13 网站地图14 网站地图15 网站地图16 网站地图17 网站地图18 网站地图19 网站地图20 网站地图21 网站地图22/a> 网站地图23