码迷,mamicode.com
首页 > 其他好文 > 详细

2015华为机试——整数分割

时间:2015-07-14 11:38:06      阅读:994      评论:0      收藏:0      [点我收藏+]

标签:2015华为机试   java   算法   math   动态规划   

题目描述:

一个整数可以拆分为2的幂的和,例如:
7 = 1+ 2 + 4
7 = 1 + 2 + 2 + 2
7 = 1 + 1 + 1 + 4
7 = 1 + 1 + 1 + 2 + 2
7 = 1 + 1 + 1 + 1 + 1 + 2
7 = 1 + 1 + 1 + 1 + 1 + 1 + 1
总共有六种不同的拆分方式
再比如: 4可以拆分成: 4 = 4, 4 = 1+1+1+1, 4 = 2+2, 4 = 1+1+2.
用f(n)表示n的不同拆分的种数,例如f(7) = 6.
要求读入n(不超过1000000),输出f(n) % 1000000000。


解题思路:

对于奇数 n = 2k+1, 它的拆分的第一项一定是1, 考虑去掉这个1,其实就一一对应于2k的拆分,因此f(2k+1) = f(2k)。
对于偶数n = 2k: 考虑有1和没有1的拆分。有1的拆分,与(2k-1)的拆分一一对应,与上面奇数的情况理由相同:没有1的拆分,将每项除以2,正好一一对应于k的所有拆分,因此f(2k) = f(2k-1) + f(k)。
最终结果只要求除以十亿的余数,在int的表示范围内,因此也不需要大数运算。注意余数的性质:(a+b)%m = (a%m+b%m)%m,所以只要对每个中间结果也都取余数,就不会有溢出问题,且不会改变最终输出结果。


代码如下:

	public static void main(String[] args)
	{
		Scanner sc=new Scanner(System.in);
		while (sc.hasNext())
		{
			int input=sc.nextInt();
			FenGe(input);
		}
		sc.close();
	}
	
	public static void FenGe(int n)
	{
		List<Integer> list=new ArrayList<Integer>();
		list.add(0);
		if ((n>=1)&&(n<=1000000))
		{
			for (int i = 1; i <= n; i++)
			{
				if (i%2==1)
				{
					if (i==1)
					{
						list.add(1);
					}
					else
					{
						list.add(list.get(i-1));
					}
				}
				
				if (i%2==0)
				{
					list.add((list.get(i-1)+list.get(i/2))%100000000);
				}
			}
			System.out.println(list.get(n));
		}
		else
		{
			System.out.println(-1);
		}
	}

版权声明:本文为博主原创文章,未经博主允许不得转载。

2015华为机试——整数分割

标签:2015华为机试   java   算法   math   动态规划   

原文地址:http://blog.csdn.net/zzc8265020/article/details/46873899

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!