60. 排列序列

解题思路

首先,排列序列呈现出明显的分段模式,如图
![](https://raw.githubusercontent.com/dendyikbc/PicGoBed/master/commonQuery/图解lc60-1 分段示意drawio.png)

对于n的序列(本质为全排列树的叶子节点)
1)segment_1类型的段 有 n 个,每段有(n-1)!;

2)对段内持续分段,直至未使用元素仅存一个时,到达最小段划分;

3)每段内的最后一个序列,保持特点:下一元素为剩余元素最大数;

 实例说明:

一级分段,segment_1,分段步长 3!= 6;

二级分段,segment_1_1,分段步长2!= 2;

三级分段,segment_1_1_1,分段步长 1!= 1;

对于segment_1,均为以1起始序列,段尾序列1432,1的下一元素为剩余元素最大数4;
对于segment1_1,均为以12起始序列,段尾序列1243,12的下一元素为剩余元素最大数4;

所以很自然想到取模进行分段选择,留余进行递推到定位到最小段(即一个叶子节点)

//伪代码
while(没定位到叶子节点){
a = k/step;
b = k%step;
//......(特殊逻辑选择当前元素)
k=b;
}


//a为模 b为余 step 为每一次分段步长
/*
特殊逻辑:
1) a=0,b!=0时 需要找到此时第1小数;
    a为0,b非0,表示本次分段步长比k大,第k个序列落在本次分段内,当前元素选择剩余最小数;
2) a=0,b=0时 需要找到此时剩余最大数;
    a为0,b为0,递推k为0,表示前一次分段刚好完整结束,第k个序列命中当前段尾,当前元素选择剩余最大数;
3) a!=0,b!=0时 找第a+1小数
    b非0,表示本次分段步长不够,需跨入下一段,当前元素选择剩余第a+1小数;
4) a!=0,b=0时 找第a小数
    b为0,表示本次分段刚好完整结束,不能跨入下一段,当前元素选择剩余第a小数;
*/

分段选取的逻辑有没有看着很复杂?xdm

这里主要是 k自减1 的引入

上述逻辑简化,首先3)是覆盖1)的;
其次,对于4)本次分段正好完整结束,k自减1 的操作,正好可以使原来的4)退化为 3),
举例,n=4,k=4,如图过程。
到这里,1)、3)、4)得到统一,重点看2)。

举例,n=4,k=4,如图过程。
步骤3,出现a=b=0,即上一步后递推k为0,前一次分段刚好完整结束,命中当前段尾,当前元素理应选择剩余最大数,
k-1引入后,对于步骤2整除现象,递推k = 2!-1,它的影响在步骤3中呈现为a =(2!-1)/1! = 2 - 1 ,注意减1的出现,在步骤3以a+1原则去寻找剩余2个数中第a+1小数时,变为寻找最大数。

对于原步骤m,出现整除现象时,k-1引入后,步骤m,a减1,b = (n-m)!-1, 它的影响在步骤m+1中呈现为a = ((n-m)!-1)/(n-m-1)! = n-m-1,在第m+1步骤中以a+1原则去寻找剩余的n-m个数字中第n-m小数,即为寻找最大数,与2)匹配。

注意:(n!-k)/(n-1)! = n - 1 ,其中 (n-1)! >= k >= 1

其实,本质上k-1的引入,在原本步骤没有整除的时候,均出现余数减1;
在整除的时候,出现商减1,余数为分段大小减1,导致的下一步骤的商与剩余数字的个数存在关系。
//伪代码
k=k-1//初始化,最前面

while(没定位到叶子节点){
a = k/step;
b = k%step;
//......(特殊逻辑选择当前元素)
k=b;
}

//a为模 b为余 step 为每一次分段步长
/*
特殊逻辑:
当前元素选择剩余第a+1小数;
*/

代码

用k做递推

class Solution {
    int n, k;
    boolean[] visit;
    // n!  t[n] n的使用 visit[n]
    int[] t = new int[]{1, 1, 2, 6, 24, 120, 720, 5040, 40320};

    public String getPermutation(int n, int k) {
        // 初始化
        this.n = n;
        this.k = k;
        visit = new boolean[n + 1];
        StringBuilder str = new StringBuilder();
        // 搜索
        for (int i = 1; i <= n; i++) {
            search(str);
        }
        return str.toString();
    }

    private void search(StringBuilder str) {
        if (str.length() == n) {
            return;
        }
        int a = k / t[n - 1 - str.length()];
        int b = k % t[n - 1 - str.length()];
        int target = 0;
        if(a==0){
            if(b!=0){
                target = 1;
            }else{
                target = n - str.length();
            }
        }else{
            if(b==0){
                target = a;
            }else{
                target = a+1;
            }
        }
        for (int i = 1; i <= n; i++) {
            if (!visit[i]) {
                --target;
                if (target == 0) {
                    str.append(i);
                    visit[i] = true;
                    break;
                }
            }
        }//fori end
        k = b;
    }
}

用k-1做递推

class Solution {
    int n, k;
    boolean[] visit;
    // n!  t[n] n的使用 visit[n]
    int[] t = new int[]{1, 1, 2, 6, 24, 120, 720, 5040, 40320};

    public String getPermutation(int n, int k) {
        // 初始化
        this.n = n;
        this.k = k - 1;
        visit = new boolean[n + 1];
        StringBuilder str = new StringBuilder();
        // 搜索
        for (int i = 1; i <= n; i++) {
            search(str);
        }
        return str.toString();
    }

    private void search(StringBuilder str) {
        if (str.length() == n) {
            return;
        }
        int a = k / t[n - 1 - str.length()] + 1;
        int b = k % t[n - 1 - str.length()];
        for (int i = 1; i <= n; i++) {
            if (!visit[i]) {
                --a;
                if (a == 0) {
                    str.append(i);
                    visit[i] = true;
                    break;
                }
            }
        }//fori end
        k = b;
    }
}

康托与逆康托

写完了去找官解的评论发现,k-1递推实际就是逆康托。

为啥k要减1?康托展开取值范围是从0开始?

解答:递推过程除法过程中余数范围为:0 ~ (n-m)!-1,前述特殊逻辑一直在处理整除情况导致的跨段与不跨段判定问题。

原题意:对n的序列编号[1,n!],取排列第k个序列,但对于第一次取模分段起,余数的范围均为[0,(n-m)!-1],0始终需特殊处理;

若转换为 对n的序列编号[0,n!-1],取排列第k-1个序列,则与余数范围存在对应,递推过程中,可以统一处理逻辑。

例如有3个数(1,2,3),则其排列组合及其相应的康托展开值如下:

排列组合名次康托展开
12310 * 2! + 0 * 1! + 0 * 0!0
13220 * 2! + 1 * 1! + 0 * 0!1
21331 * 2! + 0 * 1! + 0 * 0!2
23141 * 2! + 1 * 1! + 0 * 0!3
31252 * 2! + 0 * 1! + 0 * 0!4
32162 * 2! + 1 * 1! + 0 * 0!5