全组合

2017-06-21  本文已影响0人  yuanxiaolan

第一种方法:
/*

  1. 所有元素的全排列: ab的全排列是ab, ba(顺序相关);
  2. 所有元素的全组合: ab的全组合是a, b, ab(顺序无关);
  3. 求n个元素中选取m个元素的组合方式有哪些: abc中选2个元素的组合是ab, ac, bc;
  4. 求n个元素中选取m个元素的排列方式有哪些: abc中选2个元素的排列是ab, ba, ac, ca, bc, cb;

可以发现, 求n个元素中选取m个元素的排列方式其实是在求出n个元素中选取m个元素的组合方式后, 对每个组合组成的元素集(数组)做全排列
*/

public class Combination {
    public static void Combinationg(char[] c){
        char[] subchars = new char[c.length]; //存储子组合数据的数组
        //全组合问题就是所有元素(记为n)中选1个元素的组合, 加上选2个元素的组合...加上选n个元素的组合的和
        for (int i = 1; i <=c.length; i++) {
            combination(c, c.length, i, subchars, i);
        }
    }
    /**
     *  n个元素选m个元素的组合问题的实现. 原理如下:
     *  从后往前选取, 选定位置i后, 再在前i-1个里面选取m-1个.
     *  如: 1, 2, 3, 4, 5 中选取3个元素.
     *  1) 选取5后, 再在前4个里面选取2个, 而前4个里面选取2个又是一个子问题, 递归即可;
     *  2) 如果不包含5, 直接选定4, 那么再在前3个里面选取2个, 而前三个里面选取2个又是一个子问题, 递归即可;
     *  3) 如果也不包含4, 直接选取3, 那么再在前2个里面选取2个, 刚好只有两个.
     *  纵向看, 1与2与3刚好是一个for循环, 初值为5, 终值为m.
     *  横向看, 该问题为一个前i-1个中选m-1的递归.
     */
    public static void combination(char[] c, int n, int m, char[] subchars,int subn) {//从n个数里面选m个
        if (m == 0) { //出口
            for (int i = 0; i < subn; ++i) {
                System.out.print(subchars[i]);
            }
            System.out.println();
        } else {
            for (int i = n; i >= m; --i) { // 从后往前依次选定一个
                subchars[m - 1] = c[i - 1]; // 选定一个后
                combination(c, i - 1, m - 1, subchars, subn); // 从前i-1个里面选取m-1个进行递归
            }
        }
    }
    public static void main(String[] args){
        char[] c={'a','b','c'};
        Combinationg(c);
    }   
}

第二种方法:

public static void arranges(char[] c){
int n=c.length;
int nbit=1<<n;
for(int i=1;i<nbit;i++){
    System.out.println("组合数值"+i+"对应编码为:");
    for(int j=0;j<n;j++){
        int temp=1<<j;
        if((temp&i)!=0){
            System.out.print(c[j]);
        }
    }
    System.out.println();
}
}
上一篇 下一篇

猜你喜欢

热点阅读