整数划分类型题目--专练

1.NOI 8787:数的划分(将n划分成k个数的划分法

总时间限制: 
1000ms

内存限制: 
65536kB
描述

将整数n分成k份,且每份不能为空,任意两份不能相同(不考虑顺序)。

例如:n=7,k=3,下面三种分法被认为是相同的。

1,1,5; 1,5,1; 5,1,1;

问有多少种不同的分法。 输出:一个整数,即不同的分法。

输入
两个整数n,k (6 < n <= 200,2 <= k <= 6),中间用单个空格隔开。
输出
一个整数,即不同的分法。
样例输入
7 3
样例输出
4
提示
四种分法为:1,1,5;1,2,4;1,3,3;2,2,3。
来源
NOIP2001复赛 提高组 第二题

#include<iostream>
using namespace std;
#include<cstdio>
long long int f[201][10];
int n,k;
int main()
{
    scanf("%d%d",&n,&k);
    for(int i=1;i<=n;++i)
      for(int j=1;j<=i&&j<=k;++j)
      {
          if(j==1)
          f[i][j]=1;
          else f[i][j]=f[i-1][j-1]+f[i-j][j];
      }
    cout<<f[n][k]<<endl;
    return 0;
 } 

2. NOI7215:简单的整数划分问题(将n划分成若干个数的划分法

总时间限制: 
100ms

内存限制: 
65536kB
描述

将正整数n 表示成一系列正整数之和,n=n1+n2+…+nk, 其中n1>=n2>=…>=nk>=1 ,k>=1 。
正整数n 的这种表示称为正整数n 的划分。正整数n 的不同的划分个数称为正整数n 的划分数。

输入
标准的输入包含若干组测试数据。每组测试数据是一个整数N(0 < N <= 50)。
输出
对于每组测试数据,输出N的划分数。
样例输入
5
样例输出
7
提示
5, 4+1, 3+2, 3+1+1, 2+2+1, 2+1+1+1, 1+1+1+1+1

#include<iostream>
using namespace std;
#include<cstdio>
#define N 51
#include<cstring>
int f[N][N];
/*f[N][N]说明把N分为1--N份的划分的划分数目*/
int main()
{
    int n;
    while(cin>>n)/*注意c++中while后面加;,不会报错,但是运行时不循环*/
    {
        memset(f,0,sizeof(f));
    for(int k=1;k<=n;++k)
      for(int i=1;i<=n;++i)/*DP方程说明*/
      {
          if(k==1||i==1)
          f[i][k]=1;/*把任何数分为1分或者把1分为k份都是1*/
          if(i==k)
          f[i][k]=f[i][k-1]+1;/*当i==k的时候,(a). 划分中包含n的情况,只有一个即 { n };
              (b). 划分中不包含n的情况,这时划分中最大的数字也一定比 n 小,即 n 的所有 ( n - 1 ) 划分。*/
          if(i<k)
          f[i][k]=f[i][i];/*当i<k时,只能把i分为最多i份,所以是f[i][i].*/
          if(i>k)
          f[i][k]=f[i-k][k]+f[i][k-1];/*当i>k的时候,把i份分为k份包括:包含最大数k,就是i-k时剩下的数(其中也可能包含k),所以是f[i-k][k],二:不包含k,那就是f[i][k-1]*/
      }
        printf("%d\n",f[n][n]);
    }

    return 0;
}

代码:

3.有划分次数和没有划分次数的对比

1.没有划分次数
if(k==1||i==1)
          f[i][k]=1;/*把任何数分为1分或者把1分为k份都是1*/
          if(i==k)
          f[i][k]=f[i][k-1]+1;/*当i==k的时候,(a). 划分中包含n的情况,只有一个即 { n };
              (b). 划分中不包含n的情况,这时划分中最大的数字也一定比 n 小,即 n 的所有 ( n - 1 ) 划分。*/
          if(i<k)
          f[i][k]=f[i][i];/*当i<k时,只能把i分为最多i份,所以是f[i][i].*/
          if(i>k)
          f[i][k]=f[i-k][k]+f[i][k-1];/*当i>k的时候,把i份分为k份包括:包含最大数k,就是i-k时剩下的数(其中也可能包含k),所以是f[i-k][k],二:不包含k,那就是f[i][k-1]*/
2.有划分次数
if(j==1)
f[i][j]=1;
else f[i][j]=f[i-1][j-1]+f[i-j][j];
没有划分次数的限制,主要处理好k>i,i==k的情况,还有就是
f[i][k]代表的不再是划分k次而是划分1--k次

4.将n划分成不大于m的划分法: 

  1).若是划分多个整数可以存在相同的:

#include<iostream>
using namespace std;
#include<cstdio>
#define N 51
int f[N][N];
/*f[i][j]含义:把i在不超过j的情况下的划分数*/
int main()
{
    int n,m;
    scanf("%d%d",&n,&m);
    for(int i=0;i<=N;++i)
     {
          f[0][i]=0;f[i][0]=0;
     }
    for(int i=1;i<=n;++i)
      for(int j=1;j<=m;++j)
      {
              if(i==j)
              f[i][j]=f[i][j-1]+1;/*i==j的划分:1.把i分为超过j-1的划分数,2.把i看做一份,也就是j,所以f[0][1--m]应该是1,也就是处理一份的情况*/
              else if(i>j)
              f[i][j]=f[i][j-1]+f[i-j][j];
            else f[i][j]=f[i][i];/*当j>i的时候对于f[i][..]的划分无影响,所以不计*/ 

      }
    printf("%d\n",f[n][m]);
    return 0;
}

2).若是划分多个不同的整数:

#include<iostream>
using namespace std;
#include<cstdio>
#define N 51
int f[N][N];
/*f[i][j]含义:把i在不超过j的情况下的划分数*/
int main()
{
    int n,m;
    scanf("%d%d",&n,&m);
    for(int i=0;i<=N;++i)
     {
          f[0][i]=0;f[i][0]=0;
     }
    for(int i=1;i<=n;++i)
      for(int j=1;j<=m;++j)
      {
              if(i==j)
              f[i][j]=f[i][j-1]+1;/*i==j的划分:1.把i分为超过j-1的划分数,2.把i看做一份,也就是j,所以f[0][1--m]应该是1,也就是处理一份的情况*/
              else if(i>j)
              f[i][j]=f[i][j-1]+f[i-j][j-1];
            else f[i][j]=f[i][i];/*当j>i的时候对于f[i][..]的划分无影响,所以不计*/ 

      }
    printf("%d\n",f[n][m]);
    return 0;
}

3).总结:

            .将n划分成不大于m的划分法的做法:

              1.初始化,f[i][0]=f[0][j]=0;

               2.转移方程if i==j的时候,f[i][j]=f[i][j-1]+1

                                if i>j f[i][j]=f[i][j-1]+f[i-j][j](可重复的)  if i>j f[i][j]=f[i][j-1]+f[i-j][j-1](不可重复的)

                                if i<f f[i][j]=f[i][i];

时间: 2024-10-19 09:39:57

整数划分类型题目--专练的相关文章

整数划分类型题目

整数划分 --- 一个老生长谈的问题: 1) 练练组合数学能力. 2) 练练递归思想 3) 练练DP 总之是一道经典的不能再经典的题目: 这道好题求: 1. 将n划分成若干正整数之和的划分数. 2. 将n划分成k个正整数之和的划分数. 3. 将n划分成最大数不超过k的划分数. 4. 将n划分成若干奇正整数之和的划分数. 5. 将n划分成若干不同整数之和的划分数. 1.将n划分成不大于m的划分法:  1).若是划分多个整数可以存在相同的:  dp[n][m]= dp[n][m-1]+ dp[n-m

hdu 1028 Ignatius and the Princess III 【整数划分】

Ignatius and the Princess III                                                                                       Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submission(s): 15730    Accepted Submission(

NYOJ746——整数划分(四)

整数划分(四) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描述 暑假来了,hrdv 又要留学校在参加ACM集训了,集训的生活非常Happy(ps:你懂得),可是他最近遇到了一个难题,让他百思不得其解,他非常郁闷..亲爱的你能帮帮他吗? 问题是我们经常见到的整数划分,给出两个整数 n , m ,要求在 n 中加入m - 1 个乘号,将n分成m段,求出这m段的最大乘积 输入 第一行是一个整数T,表示有T组测试数据 接下来T行,每行有两个正整数 n,m ( 1<= n <

NYOJ-571 整数划分(三)

此题是个非常经典的题目,这个题目包含了整数划分(一)和整数划分(二)的所有情形,而且还增加了其它的情形,主要是用递归或者说是递推式来解,只要找到了递推式剩下的任务就是找边界条件了,我觉得边界也是非常重要的一步,如果找不准边界,这个题也很难做出来,当时我就是找边界找了好长时间,边界得琢磨琢磨.递推步骤如下: 第一行:将n划分成若干正整数之和的划分数.状态转移方程:dp[i][j]:和为i.最大数不超过j的拆分数dp[i][j]可以分为两种情况:1.拆分项至少有一个j 2.拆分项一个j也没有dp[i

算法笔记——整数划分3

题目来源:POJ1664-放苹果 和POJ3014 问题描述: 把m个同样的苹果放在n个同样的盘子里,允许有的盘子空着不放,问共有多少种不同的分法?(用K表示)5,1,1和1,5,1 是同一种分法. 输入: 第一行是测试数据的数目t(0 <= t <= 20).以下每行均包含二个整数m和n,以空格分开.1<=m,n<=10. 输出: 对输入的每组数据m和n,用一行输出相应的K. 分析: 问题描述转换成整数划分形式:把一个正整数m分成至多n个正整数的和,有多少种分法? 假设用f(m,

51nod p1201 整数划分

1201 整数划分 基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题 将N分为若干个不同整数的和,有多少种不同的划分方式,例如:n = 6,{6} {1,5} {2,4} {1,2,3},共4种.由于数据较大,输出Mod 10^9 + 7的结果即可. Input 输入1个数N(1 <= N <= 50000). Output 输出划分的数量Mod 10^9 + 7. Input示例 6 Output示例 4 分析:这题关键在于不同的整数一个包含数字最多的划分必

算法笔记——整数划分2

题目来源:NYOJ176 问题描述: 把一个正整数m分成n个正整数的和,有多少种分法? 例:把5分成3个正正数的和,有两种分法: 1 1 3 1 2 2 输入: 第一行是一个整数T表示共有T组测试数据(T<=50) 每组测试数据都是两个正整数m,n,其中(1<=n<=m<=100),分别表示要拆分的正数和拆分的正整数的个数. 输出: 输出每组拆分的方法的数目. 分析: 题目可以换种等价描述:把m个同样的苹果放在n个同样的盘子里,不允许有的盘子空着不放,问共有多少种不同的分法,其中n

算法笔记——整数划分1

题目来源:NYOJ90 问题描述: 将正整数n表示成一系列正整数之和:n=n1+n2+…+nk, 其中n1≥n2≥…≥nk≥1,k≥1. 正整数n的这种表示称为正整数n的划分.求正整数n的不 同划分个数.  例如正整数6有如下11种不同的划分:  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. 输入: 第一行是测试数据的数目M(1<=M<=10).以下每行均包含一个整数n(1<

整数划分 (区间DP)

整数划分(四) 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描述 暑假来了,hrdv 又要留学校在参加ACM集训了,集训的生活非常Happy(ps:你懂得),可是他最近遇到了一个难题,让他百思不得其解,他非常郁闷..亲爱的你能帮帮他吗? 问题是我们经常见到的整数划分,给出两个整数 n , m ,要求在 n 中加入m - 1 个乘号,将n分成m段,求出这m段的最大乘积 输入 第一行是一个整数T,表示有T组测试数据接下来T行,每行有两个正整数 n,m ( 1<= n <