洛谷P1028

题目描述

我们要求找出具有下列性质数的个数(包含输入的自然数n):

先输入一个自然数n(n<=1000),然后对此自然数按照如下方法进行处理:

1.不作任何处理;

2.在它的左边加上一个自然数,但该自然数不能超过原数的一半;

3.加上数后,继续按此规则进行处理,直到不能再加自然数为止.

输入输出格式

输入格式:
一个自然数n(n<=1000)

输出格式:
一个整数,表示具有该性质数的个数。

输入输出样例

输入样例#1:
6
输出样例#1:
6


说明

满足条件的数为

6,16,26,126,36,136

 

分析:

之前通过递归来做,结果时间出问题,所以通过递推来做,所以做题的思路主要是找到递推公式,然后建立数组从1推到需要的数字就可以解决

 

手动模拟一下,找一下规律

n=0,n=1时,答案显然是1
n=2, ans=2;    n=3,ans=2
n=4,ans=4;    n=5,ans=4
n=6,ans=6;    n=7,ans=6

相信大家也发现了,2n与2n+1(n为非负整数)的答案是一样的 这就是第一个规律

然后我们以n=8为例,手动模拟一下

 

然后我们以n=8为例,手动模拟一下

一共有10组解

8       1 8       2 8               3 8                          4 8

                   1 2 8             1 3 8             1 4 8         2 4 8

                                                                             1 2 4 8

 

我打出的东西很像一棵搜索树。。。

当我们把8和8下面的左三棵子树放在一起(即8和下面三列),并将所有的8都改成7,我们能发现,我们得到了n=7时的所有解;

我们再把最右端的子树(即剩下的部分)中的所有8删去,我们得到了n=4时的所有解

就这样,我们可以得到一个递推式,

    f(n)=f(n-1)                //7=8-1

        +f(n/2)                //4=8/2

再结合之前发现的规律

就能得到:

n%2==0时
    f(n)=f(n-1)+f(n/2)
n%2==1时
    f(n)=f(n-1)
然后问题就迎刃而解啦

设f[i]为初始值为i时的满足条件总数,可得f[i]=f[1]+f[2]+f[3]+...+f[i/2];容易想到f[1]=1;

因为f[i]=f[1]+f[2]+f[3]+...+f[i/2] 所以当i为奇数时f[i]=f[i-1],当i为偶数时f[i]=f[i-1]+f[i/2];

import java.util.Scanner;

public class P1028 {
    
    public static int[] f = new int[1001];
    
    public static void main(String []args){
        Scanner cin=new Scanner(System.in);
        int n=cin.nextInt();
        f[1]=1;
        for(int i=2;i<=n;i++)
        {
            f[i]=f[i-1];
            if(i%2==0)
                f[i]+=f[i/2];
        }
        System.out.println(f[n]);
    }
}
  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值