[Leetcode][第60题][JAVA][第k个排列][回溯][DFS][剪枝]

忘是亡心i 2022-12-04 02:27 210阅读 0赞

【问题描述】[中等]

在这里插入图片描述

【解答思路】

1. 回溯搜索算法 + 剪枝 ,直接来到叶子结点

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

时间复杂度:O(N^2) 空间复杂度:O(N)
在这里插入图片描述

  1. import java.util.Arrays;
  2. public class Solution {
  3. /** * 记录数字是否使用过 */
  4. private boolean[] used;
  5. /** * 阶乘数组 */
  6. private int[] factorial;
  7. private int n;
  8. private int k;
  9. public String getPermutation(int n, int k) {
  10. this.n = n;
  11. this.k = k;
  12. calculateFactorial(n);
  13. // 查找全排列需要的布尔数组
  14. used = new boolean[n + 1];
  15. Arrays.fill(used, false);
  16. StringBuilder path = new StringBuilder();
  17. dfs(0, path);
  18. return path.toString();
  19. }
  20. /** * @param index 在这一步之前已经选择了几个数字,其值恰好等于这一步需要确定的下标位置 * @param path */
  21. private void dfs(int index, StringBuilder path) {
  22. if (index == n) {
  23. return;
  24. }
  25. // 计算还未确定的数字的全排列的个数,第 1 次进入的时候是 n - 1
  26. int cnt = factorial[n - 1 - index];
  27. for (int i = 1; i <= n; i++) {
  28. if (used[i]) {
  29. continue;
  30. }
  31. if (cnt < k) {
  32. k -= cnt;
  33. continue;
  34. }
  35. path.append(i);
  36. used[i] = true;
  37. dfs(index + 1, path);
  38. // 注意 1:没有回溯(状态重置)的必要
  39. // 注意 2:这里要加 return,后面的数没有必要遍历去尝试了
  40. return;
  41. }
  42. }
  43. /** * 计算阶乘数组 * * @param n */
  44. private void calculateFactorial(int n) {
  45. factorial = new int[n + 1];
  46. factorial[0] = 1;
  47. for (int i = 1; i <= n; i++) {
  48. factorial[i] = factorial[i - 1] * i;
  49. }
  50. }
  51. }
  52. 作者:liweiwei1419
  53. 链接:https://leetcode-cn.com/problems/permutation-sequence/solution/hui-su-jian-zhi-python-dai-ma-java-dai-ma-by-liwei/
2. 有序数组(链表)模拟

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

时间复杂度:O(N2) 空间复杂度:O(N)

  1. import java.util.LinkedList;
  2. import java.util.List;
  3. public class Solution {
  4. public String getPermutation(int n, int k) {
  5. // 注意:相当于在 n 个数字的全排列中找到下标为 k - 1 的那个数,因此 k 先减 1
  6. k --;
  7. int[] factorial = new int[n];
  8. factorial[0] = 1;
  9. // 先算出所有的阶乘值
  10. for (int i = 1; i < n; i++) {
  11. factorial[i] = factorial[i - 1] * i;
  12. }
  13. // 这里使用数组或者链表都行
  14. List<Integer> nums = new LinkedList<>();
  15. for (int i = 1; i <= n; i++) {
  16. nums.add(i);
  17. }
  18. StringBuilder stringBuilder = new StringBuilder();
  19. // i 表示剩余的数字个数,初始化为 n - 1
  20. for (int i = n - 1; i >= 0; i--) {
  21. int index = k / factorial[i] ;
  22. stringBuilder.append(nums.remove(index));
  23. k -= index * factorial[i];
  24. }
  25. return stringBuilder.toString();
  26. }
  27. }
  28. 作者:liweiwei1419
  29. 链接:https://leetcode-cn.com/problems/permutation-sequence/solution/hui-su-jian-zhi-python-dai-ma-java-dai-ma-by-liwei/

【总结】

1. 剪枝大法好 充分利用条件边界 减少回溯
2.回溯思想

【数据结构与算法】【算法思想】回溯算法

3.常规思路超时 全排列 dfs传递的是深度 不是具体某一个值

相关题目
[Leedcode][JAVA][第46题][全排列][回溯算法]

  1. class Solution {
  2. private static String str = "";
  3. public String getPermutation(int n, int k) {
  4. List<String> a =new ArrayList<String>();
  5. boolean[] used = new boolean[n];
  6. int[]num = new int[n];
  7. for(int i =0 ;i<n;i++){
  8. num[i] = i+1;
  9. }
  10. permunte(0,n,num,a,used,k,"");
  11. return str;
  12. }
  13. private void permunte(int depth, int n,int[] num, List<String> a,boolean[] used ,int k,String s){
  14. if(s.length() == n){
  15. a.add(s);
  16. if(a.size() == k){
  17. str = s;
  18. return ;
  19. }
  20. }
  21. for(int i=0;i<n;i++){
  22. StringBuffer sb = new StringBuffer(s);
  23. if(!used[i]){
  24. used[i] = true;
  25. sb.append(num[i]);
  26. permunte(depth+1,n,num,a,used,k,sb.toString());
  27. used[i] = false;
  28. sb.deleteCharAt(sb.length()-1);
  29. }
  30. }
  31. }
  32. }

转载链接:https://leetcode-cn.com/problems/permutation-sequence/solution/hui-su-jian-zhi-python-dai-ma-java-dai-ma-by-liwei/

发表评论

表情:
评论列表 (有 0 条评论,210人围观)

还没有评论,来说两句吧...

相关阅读