Problem 1036 - 神奇的盒子

/*

Time Limit: 1000MS   Memory Limit: 65536KB   Difficulty:
Total Submit:
249  Accepted: 107  Special Judge: No



Description


  WM平时喜欢在夜晚游历于花丛、土堆等新校区隐蔽的角落,因为他觉得那样可以见到某些平时见不到的东西,找到某些比较稀奇的物品。随着其长期的寻觅,他终于找到了一个外表精美但上了锁的盒子,盒子没有锁眼,只有刻在上面的3对半数字:(1,1)、(4,5)、(7,15)和那个独立的数字100。

WM将盒子带回实验室,和ZYF一起研究,经过商议与复杂的推算,他们认定这些以(x,y)形式出现的数值就是传说中的拆分数(即将x变为一系列数连加,而不同的加法式子的数目就是其拆分数y),只要得到100的拆分数就能打开这个盒子。

例如,对于3,其不同的连加式有3个,即:3 = 1 + 1 + 1,3 = 1 + 2,3 = 3。式子1 + 2与式子2 + 1视为相同的式子。

WM和ZYF决定编写一个程序来计算给定正整数的拆分数,从而可以打开这个盒子,看看里头到底有什么秘密。

Input

输入数据的第一行是一个正整数T(1≤T≤100),表示有T组待测数据,每组待测数据占一行,是一个正整数N(1≤N≤100)。

Output

对于每组输入数据,输出一个整数表示N的拆分数,输出每个整数后换行。

Sample
Input

3
1
4
7

Sample Output

1
5
15

*/

整数划分问题,主要是分情况讨论,递归的思路加上dp避免重复计算,提高效率。不懂整数划分的可以自行百度。

 1 #include<stdio.h>
 2 int dp[110][110];
 3 int main()
 4 {
 5     int i,j,n,T;
 6     for(i=1;i<=100;i++) {dp[i][1]=1;dp[1][i]=1;}    //回归条件
 7     for(i=2;i<=100;i++)
 8     {
 9         for(j=2;j<=100;j++)
10         {
11             if(i<j) dp[i][j]=dp[i][i];  //对i最大数为j的整数划分的三种情况
12             if(i==j) dp[i][j]=1+dp[i][j-1];
13             if(i>j) dp[i][j]=dp[i-j][j]+dp[i][j-1];  //分包含j和不包含j
14         }
15     }
16     scanf("%d",&T);
17     while(T--)
18     {
19         scanf("%d",&n);
20         printf("%d\n",dp[n][n]);
21     }
22     return 0;
23 }

 

posted @ 2013-08-11 09:00  hjf007  阅读(246)  评论(0编辑  收藏  举报