算法训练(leetcode)第二十二天 | 491. 非递减子序列、全排列、47. 全排列 II

刷题记录

  • [491. 非递减子序列](#491. 非递减子序列)
  • [46. 全排列](#46. 全排列)
  • [47. 全排列 II](#47. 全排列 II)

491. 非递减子序列

leetcode题目地址

本题对于去重是一个难点,因为题目不允许排序,所以需要加一个笔记数组来判断相同的元素在同一层是否已经使用。使用set、map都可以达到这个目的。

时间复杂度: O ( n ∗ 2 n ) O(n * 2^n) O(n∗2n)
空间复杂度: O ( n ) O(n) O(n)

cpp 复制代码
// c++
class Solution {
public:
    vector<int> cur;
    void backtracking(vector<vector<int>> &result, const vector<int>& nums, int left){
        if(cur.size()>1){
            result.emplace_back(cur);
        }
        // 负责本层的去重
        unordered_set<int> uset;
        for(int i=left; i<nums.size(); i++){
            // 去重
            if(uset.find(nums[i])!=uset.end()) continue;
            if(cur.size()==0 || cur[cur.size()-1] <= nums[i]) {
                cur.emplace_back(nums[i]);
                uset.insert(nums[i]);
                backtracking(result, nums, i+1);
                cur.pop_back();
            }

        }
    }
    vector<vector<int>> findSubsequences(vector<int>& nums) {
        vector<vector<int>> result;
        backtracking(result, nums, 0);
        return result;
    }
};

46. 全排列

leetcode题目地址

全排列无需记录起始位置(每层都是从头开始找未使用过的元素),只需要控制每个元素在每个排列中只使用一次。借助一个额外的数组或map来实现。

时间复杂度: O ( n ! ) O(n!) O(n!)
空间复杂度: O ( n ) O(n) O(n)

cpp 复制代码
// c++
class Solution {
public:
    vector<int> cur;
    unordered_map<int, int> used;
    
    void backtracking(vector<vector<int>>&result, const vector<int>& nums){
        if(cur.size()==nums.size()){
            result.emplace_back(cur);
            return;
        }
        for(int i=0; i<nums.size(); i++){
            if(!used[i]){
                cur.emplace_back(nums[i]);
                used[i] = 1;
                backtracking(result, nums);
                used[i] = 0;
                cur.pop_back();
            }
            
        }
    }
    vector<vector<int>> permute(vector<int>& nums) {
        vector<vector<int>> result;
        backtracking(result, nums);
        return result;
    }
};

47. 全排列 II

leetcode题目地址

本题在上一题的基础上加入了相同的元素,因此需要对相同元素发起的全排列进行去重,有两种写法,一种是借助一个数组或容器来标识当前层是否已经使用了与当前元素相同的元素;另一种是排序过后判断前一个搜索过的元素与当前元素是否相同。

时间复杂度: O ( n ! ∗ n ) O(n! * n) O(n!∗n)
空间复杂度: O ( n ) O(n) O(n)

去重写法一

cpp 复制代码
// c++
class Solution {
public:
    vector<int> cur;
    // 纵向 标识单个排列中当前元素是否使用
    unordered_map<int, int> used;
    void backtracking(vector<vector<int>>&result, const vector<int>& nums){
        if(cur.size() == nums.size()){
            result.emplace_back(cur);
            return;
        }
        // 横向 标识所有排列中与当前元素相同的值是否已经参加过查找
        unordered_set<int> duplicate;
        for(int i=0; i<nums.size(); i++){
            if(!used[i] && duplicate.find(nums[i])==duplicate.end()){
                used[i] = 1;
                duplicate.insert(nums[i]);
                cur.emplace_back(nums[i]);
                backtracking(result, nums);
                cur.pop_back();
                used[i] = 0;
            }
        }
    }
    vector<vector<int>> permuteUnique(vector<int>& nums) {
        vector<vector<int>> result;
        backtracking(result, nums);
        return result;
    }
};

去重写法二

cpp 复制代码
// c++
class Solution {
public:
    vector<int> cur;
    // 纵向 标识单个排列中当前元素是否使用
    unordered_map<int, int> used;
    void backtracking(vector<vector<int>>&result, const vector<int>& nums){
        if(cur.size() == nums.size()){
            result.emplace_back(cur);
            return;
        }
    
        for(int i=0; i<nums.size(); i++){
           
            if(i>0 && nums[i]==nums[i-1] && used[i-1]) continue;
            if(!used[i]){
                used[i] = 1;
                cur.emplace_back(nums[i]);
                backtracking(result, nums);
                cur.pop_back();
                used[i] = 0;
            }
        }
    }
    vector<vector<int>> permuteUnique(vector<int>& nums) {
        sort(nums.begin(), nums.end()); // 排序
        vector<vector<int>> result;
        backtracking(result, nums);
        return result;
    }
};
相关推荐
A7bert777几秒前
【YOLOv8pose部署至RDK X5】模型训练→转换bin→Sunrise 5部署
c++·python·深度学习·yolo·目标检测
生成论实验室13 分钟前
《事件关系阴阳博弈动力学:识势应势之道》第二篇:阴阳博弈——认知的动力学基础
数据结构·人工智能·科技·神经网络·算法
li16709027025 分钟前
第二十七章:智能指针
c语言·数据结构·c++·visual studio
风筝在晴天搁浅42 分钟前
字节高频题 小于n的最大数
算法
LabVIEW开发44 分钟前
LabVIEW水力机组空蚀在线监测
算法·labview·labview知识·labview功能·labview程序
AI科技星1 小时前
科幻艺术书本封面:《全域数学》第一部·数术本源 第三卷 代数原本(P95-141)完整五级目录【乖乖数学】
算法·机器学习·数学建模·数据挖掘·量子计算
风筝在晴天搁浅1 小时前
LeetCode 92.反转链表Ⅱ
算法·leetcode·链表
王老师青少年编程1 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【贪心与二分判定】:数列分段 Section II
c++·算法·贪心·csp·信奥赛·二分判定·数列分段 section ii
zh_xuan1 小时前
libcurl调用https接口
c++·libcurl
就叫飞六吧1 小时前
QT写一个桌面程序exe并动态打包基本流程(c++)
开发语言·c++