欢迎访问 生活随笔!

生活随笔

当前位置: 首页 > 编程资源 > 编程问答 >内容正文

编程问答

半数集问题

发布时间:2023/11/29 编程问答 37 豆豆
生活随笔 收集整理的这篇文章主要介绍了 半数集问题 小编觉得挺不错的,现在分享给大家,帮大家做个参考.

  给定一个自然数n,由n开始可以依次产生半数集set(n)中的数如下:

  (1)   n ∈set(n);

  (2) 在n的左边加上一个自然数,但该自然数不能超过最近添加的数的一半;

  (3) 按此规则进行处理,直到不能再添加自然数为止。

  以6为例子,6,6前面可以加1,2,3生成16,26,36,26前面可以加1生成126,同理36生成136.所以6的半数集元素个数为6分别是6,16,26,36,126,136

  以12为例子,只加一个数字产生的元素有612,512,412,312,212,112。因为之后加的数字与‘12’没有关系,只与第一次加的数字有关,612,512,412,312,212,112产生的半数集元素的个数相当于6,5,4,3,2,1的半数集的个数,不难得到如下公式。

  

递归代码:

#include <algorithm> #include <iostream> #include <cstring> #include <vector> #include <cstdio>using namespace std;int f(int n){if(n==1)return 1;else{int ans=1;for(int i=1;i<=n/2;i++){ans=ans+f(i);}return ans;} }int main() {int n;scanf("%d",&n);int num=f(n);printf("%d\n",num);return 0; }

  递归时间复杂度很高对此我们可以进行优化,用数组空间存储之前的状态

O(n2)实现:

#include <algorithm> #include <iostream> #include <cstring> #include <vector> #include <cstdio>using namespace std;int main() {int n;scanf("%d",&n);int a[100];a[1]=1;for(int i=2;i<=n;i++){a[i]=1;for(int j=1;j<=i/2;j++){a[i]+=a[j];}}printf("%d\n",a[n]);return 0; }

  仔细分析后,发现时间复杂度其实可以降到O(n),因为当n为奇数时,第n项的半数集个数等于第n-1项的半数集个数,当n为偶数时,第n项的半数集个数等于第n-1项半数集个数 加 第n/2项半数集个数。

O(n)实现:

#include <algorithm> #include <iostream> #include <cstring> #include <vector> #include <cstdio>using namespace std;int main() {int n;scanf("%d",&n);int a[100];a[0]=a[1]=1;for(int i=2;i<=n;i++){if(i%2!=0)a[i]=a[i-1];else{a[i]=a[i-1]+a[i/2];}}printf("%d\n",a[n]);return 0; }

  

转载于:https://www.cnblogs.com/wz-archer/p/10603424.html

总结

以上是生活随笔为你收集整理的半数集问题的全部内容,希望文章能够帮你解决所遇到的问题。

如果觉得生活随笔网站内容还不错,欢迎将生活随笔推荐给好友。