整数划分(一)
n=m1+m2+…+mi; (其中mi为正整数,并且1 <= mi <= n),则{m1,m2,…,mi}为n的一个划分。
如果{m1,m2,…,mi}中的最大值不超过m,即max(m1,m2,…,mi)<=m,则称它属于n的一个m划分。这里我们记n的m划分的个数为q(n,m);
根据m和n的关系可将问题划分为:
1.n或m<0时, q(n,m)=0;
2.n或m等于0时, q(n,m)=1;例如q(n,1)只有n一种情况,q(1,m)也只有1一种情况
3.n小于m时,这时q(n,m)=q(n,n),因为由于划分中不可能出现负数。
4.n等于m时,q(n,m)=q(n,m-1)+1
5.n>m时,q(n,m)=q(n,m-1)+q(n-m,m)。q(n,m-1)相当于不包含m的划分,q(n-m,m)相当于包含m的划分。
#include<iostream>
using namespace std;
int q(int n,int m)
{
if(n==0||m==0) return 0;
else if(n==1||m==1) return 1;
else if(n<m) return q(n,n);
else if(n==m) return q(n,m-1)+1;
else if(n>m) return q(n,m-1)+q(n-m,m);
}
int main()
{
int n;
cin>>n;
int ans=q(n,n);
cout<<ans<<endl;
return 0;
}
整数划分(二)
题目:把一个正整数n分成m个正整数的和,有多少种分法?
首先拿6来说,如果要分成3个数相加,可分为 (1,1,4)(1,2,3),(2,2,2)
可以看出,可分为一组有1,一组不包含1
有1的一组我们把1除掉以便进行递归f(n,m)=f(n-1,m-1)
没有1的一组,我们将每个数都减1,(2,2,2)就变成了(1,1,1)也就成为包含1的组合了,递归式就出来了。f(6,3)=f(5,2)+f(3,3)
同时我们考虑到当n=0或n<m时,f(n,m)=0,m=1或者n=m时,f(n,m)=1;
#include<iostream>
using namespace std;
int f(int n,int m)
{
if(n==0||n<m) return 0;
else if(n==m||m==1) return 1;
else return f(n-1,m-1)+f(n-m,m);
}
int main()
{
int n,m;
cin>>n>>m;
cout<<f(n,m)<<endl;
return 0;
}