半数集问题和半数集单集问题

半数集问题描述
给定一个自然数n,由n 开始可以依次产生半数集set(n)中的数如下。
(1) n∈set(n);
(2) 在n 的左边加上一个自然数,但该自然数不能超过最近添加的数的一半;
(3) 按此规则进行处理,直到不能再添加自然数为止。
例如,set(6)={6,16,26,126,36,136}。半数集set(6)中有6 个元素。
注意半数集是多重集。

其实很容易想到使用递归实现,算法如下:

import java.util.Scanner;

/**
 * DateTime: 2016/10/6 11:18
 * 功能:半数集问题
 * 思路:
 */
public class Topic2_3 {

    public static void main(String[] args) {
        Scanner in=new Scanner(System.in);
        int n=in.nextInt();//要分解的数
        System.out.println(Fun(n));
    }

    public static int Fun(int n) {
        int result=1;
        if(n>1){
            for ( int i = 1; i <= n/2 ; i++ ) {
                result +=Fun(i);
            }
        }
        return result;
    }

}

但是仔细观察上面的式子就会发现递归有一个缺陷就是在重复的计算已经使用过的值,这样会加大空间和时间的开销,因此我们可以用数组把计算过的值存起来,下次直接取。
算法如下:

import java.util.Scanner;

/**
 * DateTime: 2016/10/6 11:18
 * 功能:半数集问题
 * 思路:
 */
public class Topic2_3 {

    private static int[] array=new int[200];//存储计算过的值

    public static void main(String[] args) {
        Scanner in=new Scanner(System.in);
        int n=in.nextInt();//要分解的数
        array[1]=1;//设置初值
        System.out.println(Fun(n));
    }

    public static int Fun(int n) {
        int result=1;
        if(array[n]>0)  return array[n];
        for ( int i = 1; i <= n/2 ; i++ ) {
            result +=Fun(i);
        }
        array[n]=result;
        return result;
    }

}


半数单集类似半数集,区别在于:半数集是多重集,而半数单集不是多重集,即集合中已有的元素不再添加到集合中。
例如:n=24,那么半数集set(24)中的元素1224就有如下两种方式可以生成:
24 → 1224
24 →224 → 1224
所以,1224就是一个被重复计算的元素。

import java.util.Scanner;

/**
 * DateTime: 2016/10/6 14:43
 * 功能:半数单集问题
 * 思路:本题的半数集是单集,不允许重复元素,因此在计算时要踢除重复的元素
 * 题目中的条件0<n<201 => 0<n/2<100 因此在计算时可能产生重复的元素是两位数
 * 2位数x重复产生的条件是:在一位数y=x%10的半数集中已产生x,因此x/10<=y/2 或等价的2*(x/10)<=x%10
 */
public class Topic2_4 {
    private static int[] array=new int[200];//存储计算过的值

    public static void main(String[] args) {
        Scanner in=new Scanner(System.in);
        int n=in.nextInt();//要分解的数
        array[1]=1;//设置初值
        System.out.println(Fun(n));
    }

    public static int Fun(int n) {
        int result=1;
        if(array[n]>0)  return array[n];
        for ( int i = 1; i <= n/2 ; i++ ) {
            result +=Fun(i);
            if((i>10) && (2*(i/10)<=i%10))
                result -=array[i/10];
        }
        array[n]=result;
        return result;
    }

}
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值