zoukankan      html  css  js  c++  java
  • (转的)计算组合数——整数拆分

     1.计算组合数: 一般常用C(n+1,k)=(n+1)!/k!/(n+1-k)!来计算组合数,但是这个方法中涉及到阶乘运算,数据n不能太大。用下面的方法则可以避免这个问题。

    帕斯卡恒等式为C(n+1,k)=C(n,k-1)+C(n,k)

    #include <stdio.h>
    #include <stdlib.h>
    void error()
    {
    printf("Something is wrong,Please check it!");
    exit(1);
    }

    int fun(int n,int m)
    {
       if(n<0||m<0)
         error();
       if(n<m) return 0;
       if(m==0||m==n)
         return 1;
       else
         return fun(n-1,m)+fun(n-1,m-1);
    }

    int main()
    {
       int n,m;
       printf("Input two data:\n");
       scanf("%d%d",&n,&m);
       printf("The combo number is:\n%d\n",fun(n,m));
       return 0;
    }

    前面的代码还是有点麻烦,递归次数太多而影响效率,例如要计算C(5,1),按照帕斯卡公式应该拆分如下:

    C(5,1) =C(4,1)+C(4,0)=C(3,1)+C(3,0)+C(4,0)=C(2,1)+C(2,0)+C(3,0)+C(4,0)=C(1,1)+C(1,0)+C(2,0)+C(3,0)+C(4,0)=5

    可以想象,如果把其中的代码写成下面的样子应该会好一些:

    long fun(long n,long r)
    {
       if(r<0||n<0)//异常处理
       {
          printf("\nError.Exit!");
          exit(-1);
       }
       if(r>n)//数学定义
          return 0;
       if(r==1||r==n-1)//根据组合定义,我们有C(n,1)=C(n,n-1)=n
          return n;
       if(r==0||r==n)//根据组合定义,我们有C(n,0)=C(n,n)=1
          return 1;
       return fun(n-1,r)+fun(n-1,r-1);//帕斯卡组合公式
    }

    如果按照这个思路延续下去,我们完全可以在代码中写出if(r==2||r==n-2)、if(r==3||r==n-3)甚至更高,但这样的话,就变成数学问题了,就不是程序设计了。最后我们再看看组合数的定义公式C(n,r)=n!/r!/(n-r)!,改写一下:

    C(n,r)=n×(n-1)×(n-2)×...×2×1/[r×(r-1)×(r-2)×...×2×1]/[(n-r)×(n-r-1)×(n-r-2)×...×2×1]

    可以看出,有些数如n到max(r,n-r)只有分子上有,有些数如max(r,n-r)到min(r,n-r)分子分母都有,而另一些数如min(r,n-r)到1分子上出现一次而分母上出现两次。因而,我把代码写成下面的样子:

    long max(long m,long n)
    {
       if(m>n)
          return m;
       return n;
    }

    long min(long m,long n)
    {
       if(m<n)
          return m;
       return n;
    }

    long newfun(long n,long r)
    {
       long i;
       long result=1;
       for(i=n;i>0;i--)
       {
          if(i>max(r,n-r))
              result*=i;
          else if(i<=min(r,n-r))
             result/=i;
       }
       return result;
    }
    写成这样是否就完美了呢?我认为还没有。仔细看一下,就会发现,在最后一个程序中多次调用max函数和min函数,很明显会影响程序效率。考虑了一下,利用“空间”来换取“时间”,我又修改成了下面的样子:

    long max(long m,long n)
    {
       if(m>n)
          return m;
       return n;
    }

    long min(long m,long n)
    {
       if(m<n)
          return m;
       return n;
    }

    long newfun(long n,long r)
    {
       long i;
       long maxvalue,minvalue;
       long result=1;
       maxvalue=max(r,n-r);
       minvalue=min(r,n-r);
       for(i=n;i>0;i--)
       {
          if(i>maxvalue)
            result*=i;
          else if(i<=minvalue)
            result/=i;
       }
       return result;
    }

    2. 计算组合:利用二进制数计算无重复组合

    #define N 5

    #i nclude <conio.h>

    void init(int arr[])
    { /* to initiate the array*/
    int i;
    for(i=0;i<N;i++)
        arr[i]=i+1;
    }

    void next(int arr[])
    { /* to compute the next possible combination*/
    int i;
    for(i=N-1;i>=0&&arr[i]==1;i--)
           arr[i]=0;
    if(i>=0)
         arr[i]=1;
    }

    int ZeroAtPos(int arr[])
    { /* return the position of 0 at the 'binary' array,if not,return -1*/
    int i;
    for(i=0;i<N;i++)
        if(arr[i]==0)
          return i;
    return -1;
    }

    void output(int arr[],int binary[])
    { /* print the current combination according to 'binary' array*/
    int i;
    for(i=0;i<N;i++)
        if(binary[i]==1)
          printf("%4d",arr[i]);
    printf("\n");
    }

    main()
    {
       int array[N];
       int binary[N]={0};
       clrscr();
       init(array);
       while(ZeroAtPos(binary)!=-1)
       {
         next(binary);
         output(array,binary);
       }
    }

    3. 计算组合:利用函数递归计算无重复组合

    int n,r;
    int a[10];

    void fun(int m,int k)
    {
       int i,j;
       for(i=m;i>=k;i--)
       {
          a[k-1]=i;
          if(k>1)
               fun(i-1,k-1);
          else
          {
             for(j=r-1;j>=0;j--)
                printf("%5d",a[j]);
             printf("\n");
          }
       }
    }

    void main()
    {
    n=5;
    for(r=1;r<=n;r++)
        fun(n,r);
    }

    4. 计算组合:利用函数递归计算有重复组合

    int r;
    int result[20]={0};
    zuhe(int m,int k)
    {
       int i,j;
       for(i=m;i>=0;i--)
       {
          result[k-1]=i;
          if(k>1)
               zuhe(i,k-1);
          else
          {
              for(j=0;j<r;j++)
                    printf("%4d",result[j]);
              printf("\n");
          }
       }
    }

    main()
    {
       r=3;
       zuhe(7,r);
    }

    5. 计算组合:常规算法计算无重复组合

    从n个数中取r个数的组合,假设c1c2c3...cr是其中一个组合,则计算其下一个组合的算法如下:

    Step 1:求满足不等式cj<n-r+j的最大坐标,并设为i。即i=max{j | cj<n-r+j}

    Step 2:设置ci=ci+1

    Step 3:设置cj=cj-1+1,对于j=i+1,i+2,i+3,...,r

    #i nclude <conio.h>

    int fun(int choice[],int r,int n)
    {//利用了上面的算法,计算下一个组合
    int i,j;
    for(i=r-1;i>=0;i--)
         if(choice[i]<n-r+i)
            break;
    if(i<0)//已经是最后一个了
         return 0;
    choice[i]+=1;
    for(j=i+1;j<r;j++)
         choice[j]=choice[j-1]+1;
    return 1;//还有后续组合
    }

    void comb(char source[],int total,int count)
    {
    int choice[5];
    int i;
    for(i=0;i<count;i++)//人工设置第一个组合
         choice[i]=i;
    do
    {
         printf("\n");
         for(i=0;i<count;i++)
         {//输出当前组合
             int t=choice[i];
              printf("%c ",source[t]);
         }
    }while(fun(choice,count,total));
    }

    void main()
    {
       char array[5]={'a','b','c','d','e'};
       clrscr();
       comb(array,5,2);
    }

    6. 组合应用:利用组合求解背包问题

         求n个元素的所有组合,有很多方法,有一个方式是这样的:定义一个参考数组并设初值为全0,然后处理参考数组(从后向前扫描,若当前元素值为1则变为0,继续扫描;若当前元素值为0则变为1,然后退出扫描)。接下来根据参考数组输出n个元素(若对应的参考数组中元素值为1,则选择此元素,即输出)。

           可以利用这个求组合的方法计算背包问题,其实背包问题本来就可以看作n个元素的r组合。代码如下:

    #define K 10
    #define N 10

     i nclude <stdlib.h>
    #i nclude <conio.h>

    void create(long array[],int n,int k)
    {
       int i,j;
       array[0]=1;
       for(i=1;i<n;i++)
       {
          long t=0;
          for(j=0;j<i;j++)
                t=t+array[j];
          array[i]=t+random(k)+1;
       }
    }
    void output(long array[],int n)
    {
       int i;
       for(i=0;i<n;i++)
       {
          if(i%5==0)
               printf("\n");
          printf("%14ld",array[i]);
       }
    }

    void beibao(long array[],int cankao[],long value,int n)
    {
       int i;
       int j;
       int cankao1[N]={0};
       long value1=0,value2=0;
       do
       {
          value2=0;
          for(j=n-1;j>=0;j--)/*处理参考数组*/
            if(cankao1[j]==0)
            {
              cankao1[j]=1;
              break;
             }
            else if(cankao1[j]==1)
                  cankao1[j]=0;
          for(i=0;i<n;i++)/*计算当前选择方案*/
             if(cankao1[i]==1)
                value2+=array[i];
          if(value2<=value&&value2>value1)/*若当前选择方案可取*/
          {
             value1=value2;
              for(i=0;i<n;i++)/*记录选择方案*/
                  cankao[i]=cankao1[i];
          }
       }while(j>=0);
    }

    void main()
    {
       long array[N];
       int cankao[N]={0};
       int i;
       long value;
       clrscr();
       create(array,N,K);
       output(array,N);
       printf("\nInput the value of beibao:\n");
       scanf("%ld",&value);
       beibao(array,cankao,value,N);
       printf("\nThe answer is:\n");
       for(i=0;i<N;i++)
          if(cankao[i]==1)
           printf("%8ld",array[i]);
    }

    7. 组合应用:利用组合求解整数拆分

         所谓整数的拆分,就是把一个整数拆成多个整数之和,如

    5=1+1+1+1+1

    5=1+1+1+2

    5=1+1+3

    5=1+2+2

    5=1+4

    5=2+3

    5=5

    其实,整数拆分问题可以用过组合来求解,也就是在C(n,0),C(n,1),...,C(n,n)所有的组合中查找组合中所有数之和为n的组合。代码如下:

    #i nclude <conio.h>

    #define N 9
    int r;
    int result[20]={0};
    zuhe(int m,int k)
    {
       int i,j;
       for(i=m;i>=1;i--)
       {
          result[k-1]=i;
          if(k>1)
               zuhe(i,k-1);
          else
          {
             int temp=0;
               for(j=0;j<r;j++)
                   temp+=result[j];
              if(temp==N)//当前组合是整数N的一个拆分
            {
                for(j=0;j<r;j++)
                    printf("%4d",result[j]);
                   printf("\n");
             }
          }
       }
    }

    main()
    {
       clrscr();
       printf("\nN=%d\n",N);
       for(r=N;r>0;r--)
          zuhe(N,r);
    }

    运行结果:


    N=7
       1   1   1   1   1   1   1
       1   1   1   1   1   2
       1   1   1   1   3
       1   1   1   2   2
       1   1   1   4
       1   1   2   3
       1   2   2   2
       1   1   5
       1   2   4
       1   3   3
       2   2   3
       1   6
       2   5
       3   4
       7
    N=6
       1   1   1   1   1   1
       1   1   1   1   2
       1   1   1   3
       1   1   2   2
       1   1   4
       1   2   3
       2   2   2
       1   5
       2   4
       3   3
       6

    N=5
       1   1   1   1   1
       1   1   1   2
       1   1   3
       1   2   2
       1   4
       2   3
       5
    N=8
       1   1   1   1   1   1   1   1
       1   1   1   1   1   1   2
       1   1   1   1   1   3
       1   1   1   1   2   2
       1   1   1   1   4
       1   1   1   2   3
       1   1   2   2   2
       1   1   1   5
       1   1   2   4
       1   1   3   3
       1   2   2   3
       2   2   2   2
       1   1   6
       1   2   5
       1   3   4
       2   2   4
       2   3   3
       1   7
       2   6
       3   5
       4   4
       8

    N=3
       1   1   1
       1   2
       3

  • 相关阅读:
    设计模-设计原则-开闭原则
    使用export/import导出和导入docker容器
    Docker学习笔记(二)
    redis-cli的一些有趣也很有用的功能
    Redis命令总结
    使用domain模块捕获异步回调中的异常
    大话程序猿眼里的高并发
    使用pm2躺着实现负载均衡
    Request —— 让 Node.js http请求变得超简单
    避免uncaughtException错误引起node.js进程崩溃
  • 原文地址:https://www.cnblogs.com/0803yijia/p/2364029.html
Copyright © 2011-2022 走看看