整数分解,输出所有分解方案

如,对于正整数n=6,可以拆分为:
6
5+1
4+2, 4+1+1
3+3, 3+2+1, 3+1+1+1
2+2+2, 2+2+1+1, 2+1+1+1+1
1+1+1+1+1+1+1
现在的问题是,对于给定的正整数n,程序输出该整数的拆分种类数(HDOJ  1028)。

DP思路:
n = n1 + n2 + n3 + n4 + .... + nk
状态表示:将n划分为k个数相加的组合方案数记为 q(n,k)。(相当于将n个苹果放入k个盘子)
状态转移:
(1)若k>n,则盘子数大于苹果数,至少有n-k个空盘子,可以将其拿掉,对组合方案数无影响。
q(n,k) = q(n,n)
(2)若k<=n,则盘子数小于等于苹果数,则分为两种情况
1.至少有一个盘子空着:q(n,k) = q(n,k-1)
2.所有盘子都不空:q(n,k) = q(n-k,k)
q(n,k) = q(n,k-1) + q(n-k,k)
方法一:DP非递归

[cpp]  view plain copy
  1. int main(void)  
  2. {  
  3.     int n,i,j,dp[121][121];  
  4.     for(i = 1 ; i < 121 ; ++i)  
  5.     {  
  6.         for(j = 1 ; j < 121 ; ++j)  
  7.         {  
  8.             if(i == 1 ||  j == 1)  
  9.                 dp[i][j] = 1;  
  10.             else if(i > j)  
  11.                 dp[i][j] = dp[i][j-1] + dp[i-j][j];  
  12.             else if(i == j)  
  13.                 dp[i][j] = dp[i][j-1] + 1;  
  14.             else  
  15.                 dp[i][j] = dp[i][i];  
  16.         }  
  17.     }  
  18.   
  19.     while(scanf("%d",&n)!=EOF)  
  20.     {  
  21.         cout<
  22.     }  
  23.     return 0;  
  24. }  
方法二:递归思路
[cpp]  view plain copy
  1. int q(int n , int m)  
  2. {  
  3.     if(n < 1 || m < 1)  
  4.         return 0;  
  5.     if(n == 1 || m == 1)  
  6.         return 1;  
  7.     if(n < m)  
  8.         return q(n , n);  
  9.     if(n == m)  
  10.         return q(n , m - 1) + 1;  
  11.     return q(n , m - 1) + q(n - m , m);  
  12. }  
  13.   
  14. int main(void)  
  15. {  
  16.     int n;  
  17.     while(scanf("%d",&n)!=EOF)  
  18.     {  
  19.         cout<
  20.     }  
  21.     return 0;  
  22. }  
#include #include using namespace std; int solve(int arr[],int left_sum,int start_index,int end_index,vector& res) { if(left_sum < 0 || start_index > end_index ) return 0; if(left_sum ==0 ) { //输出答案 int len = res.size(); for(int i=0;i= arr[start_index]) { //要 arr[start_index] 这个数 res.push_back(arr[start_index]); solve (arr,left_sum-arr[start_index],start_index+1,end_index, res) ; res.pop_back(); // 不 要 arr[start_index] 这个数 solve(arr,left_sum,start_index+1,end_index, res) ; } return 0; } int solve1(int arr[],int left_sum,int start_index,int end_index,vector& res) { if(left_sum < 0 || start_index > end_index ) return 0; if(left_sum ==0 ) { //输出答案 int len = res.size(); for(int i=0;i= arr[i]) { res.push_back(arr[i]); solve1 (arr,left_sum-arr[i],i,end_index, res) ; res.pop_back(); } else break; //提前减枝 } return 0; } int main() { int arr[11]; for(int i=1;i<=10;i++) arr[i]=i; vector res; //不能重复 solve1(arr,8,0,10,res); //能重复 solve1(arr,8,1,10,res); system("pause"); return 0; }

 

 

你可能感兴趣的:(面试题)