全排列算法-递归&字典序实现(C++)

全排列:

从n个不同元素中任取m(m≤n)个元素,按照一定的顺序排列起来,叫做从n个不同元素中取出m个元素的一个排列。当m=n时所有的排列情况叫全排列。
例如:

1、2、3三个元素的全排列为:
{1,2,3},{1,3,2},{2,1,3},{2,3,1},{3,1,2},{3,2,1}。

解法1(递归)

如下图:要对1、2、3、4进行排序,第一个位置上的元素有四种可能:1或2或3或4,假如已经确定了第一个元素为4,剩下的第二个位置上可以是1、2、3,很显然这具有递归结构,如果原始要排列的数组顺序为1、2、3、4,现在只要分别交换1、2,1、3,1、4然后对剩下的3个元素进行递归的排列。

image

代码:

    void permute(string str, int k) {
        if (k == str.size() - 1) {
            cout << str << endl;
        } else if (k < str.size() - 1) {
            for (int i = k; i < str.size(); i++) {
                swap(str[i], str[k]);
                permute(str, k + 1);
                swap(str[i], str[k]);
            }
        }
    }

    void permute(string str) {
        if (!str.empty()) {
            permute(str, 0);
        }
    }

递归方法会对重复元素进行交换比如使用递归对{1,1}进行全排序会输出:{1,1},{1,1}两个重复的结果。要在排序的时候去掉重复结果,可以修改一下代码如下:

    void permute(string str, int k) {
        if (k == str.size() - 1) {
            cout << str << endl;
        } else if (k < str.size() - 1) {
            for (int i = k; i < str.size(); i++) {
                if (i == k || str[i] != str[k]) {
                    swap(str[i], str[k]);
                    permute(str, k + 1);
                    swap(str[i], str[k]);
                }
            }
        }
    }

    void permute(string str) {
        if (!str.empty()) {
            permute(str, 0);
        }
    }
解法2(字典序法)

字典序法
对给定的字符集中的字符规定了一个先后关系,在此基础上规定两个全排列的先后是从左到右逐个比较对应的字符的先后。
列如:对a、b、c进行排序的结果是{a,b,c}、{a,c,b}、{b,a,c}、{b,c,a}、{c,a,b}、{c,b,a}
字典序法的优点是排列的结果按照顺序输出并且对于重复的元素不进行重复排序。

字典排序法的思想:

例如:对元素1,2,3,4进行排序,假设默认的数组顺序为{1,2,3,4},先输出第一个排列:1、2、3、4。然后从右向左找到第一个非递增的数,4,3,因为3比4小,交换3、4,并且对3后面的数进行逆序排列,第二个排列为{1,2,4,3},再从右向左3,4,2,发现2比4小,交换从右向左第一个比2大的数,交换后{1,3,4,2}再对3后面的数进行逆序排列第三个序列为:{1,3,2,4}
依次循环直到数组成为完全递减数组结束1、2、3、4字典排序的最大序列为{4,3,2,1}。

代码:

    void permute(string str) {
        if (!str.empty()) {
            permute(str, 0);
        }
    }

    string permute(string &src, int k) {
        sort(src.begin(), src.end());
        int len = src.length();
        bool end = false;
        while (true) {
            cout << src << endl;

            int index = 0;
            int j;

            // 从右到左找到第一个非递增的元素
            for (j = len - 2; j >= 0; j--) {
                if (src[j] < src[j + 1]) {
                    index = j;
                    break;
                } else if (j == 0) {
                    end = true;
                    break;
                }
            }

            // 遍历已经结束
            if (end) {
                break;
            }

            // 从右向左找到第一个比非递增元素大的元素
            for (j = len - 1; j >= 0; j--) {
                if (src[j] > src[index]) {
                    break;
                }
            }

            swap(src[index], src[j]);
            reverse(src, index + 1);
        }
    }

    void reverse(string &str, int index) {
        int i = index;
        int j = str.length() - 1;
        while (i < j) {
            swap(str[i], str[j]);
            i++;
            j--;
        }
    }

转载请注明出处::https://juejin.cn/post/6897762695689273351
参考文献:笔试面试算法经典--全排列算法-递归&字典序实现(Java)

你可能感兴趣的:(全排列算法-递归&字典序实现(C++))