首页 > 其他分享 >[491] Non-decreasing Subsequences

[491] Non-decreasing Subsequences

时间:2024-04-19 10:36:47浏览次数:13  
标签:Non used 递归 nums tempResult vector Subsequences result 491

算法助手

用户:

这个题目有什么好的思路吗?“Given an integer array nums, return all the different possible non-decreasing subsequences of the given array with at least two elements. You may return the answer in any order.”
我的代码是这样的:

/*
 * @lc app=leetcode.cn id=491 lang=cpp
 *
 * [491] Non-decreasing Subsequences
 */
#include "General.h"

// @lc code=start
class Solution
{
private:
    vector<vector<int>> result;

    void search(vector<int> &nums, vector<int> &tempResult, int left)
    {
        if (tempResult.size() > 1)
        {
            if (tempResult[tempResult.size() - 1] >= tempResult[tempResult.size() - 2])
            {
                result.push_back(tempResult);
            }
            else
            {
                return;
            }
        }

        for (int i = left; i < nums.size(); i++)
        {
            if (i > left && nums[i] == nums[i - 1])
            {
                continue;
            }
            tempResult.push_back(nums[i]);
            search(nums, tempResult, i + 1);
            tempResult.pop_back();
        }
    }

public:
    vector<vector<int>> findSubsequences(vector<int> &nums)
    {
        vector<int> v;
        search(nums, v, 0);
        return result;
    }
};
// @lc code=end

#include <cassert>
#include <vector>

int main(int argc, char const *argv[])
{
    Solution sol;
    vector<int> nums = {1, 2, 1, 1};
    vector<vector<int>> result = sol.findSubsequences(nums);
    cout << '[' << endl;
    for (vector<int> ve : result)
    {
        cout << "  [";
        for (int i : ve)
        {
            cout << i << ", ";
        }
        cout << "]," << endl;
    }
    cout << ']' << endl;
    return 0;
}

ChatGPT:

您的代码已经很接近正确答案了,但是存在一些问题需要解决。首先,您需要避免生成重复的子序列。在您的代码中,您尝试通过检查 if (i > left && nums[i] == nums[i - 1]) 来避免重复,但这只能在同一层递归中避免重复,如果重复的数字在不同层递归中出现,这种方法就无效了。

解决这个问题的一个方法是在每一层递归中使用一个哈希集合来记录已经使用过的数字,从而避免在同一层递归中使用重复的数字。此外,您在添加到结果集 result 之前检查 tempResult 是否是非递减的,这是不必要的,因为在添加新元素之前,您可以先检查它是否不小于 tempResult 的最后一个元素。

以下是修改后的代码:

#include <vector>
#include <set>
using namespace std;

class Solution {
private:
    vector<vector<int>> result;
    set<vector<int>> uniqueResults; // 用于记录唯一的子序列

    void search(vector<int> &nums, vector<int> &tempResult, int start) {
        if (tempResult.size() > 1) {
            result.push_back(tempResult);
        }

        set<int> used; // 用于当前递归层去重
        for (int i = start; i < nums.size(); i++) {
            // 如果当前数字已经在这一层使用过,跳过
            if (used.find(nums[i]) != used.end()) continue;
            // 如果当前数字小于tempResult中最后一个数字,跳过
            if (!tempResult.empty() && nums[i] < tempResult.back()) continue;

            used.insert(nums[i]);
            tempResult.push_back(nums[i]);
            search(nums, tempResult, i + 1);
            tempResult.pop_back();
        }
    }

public:
    vector<vector<int>> findSubsequences(vector<int>& nums) {
        vector<int> tempResult;
        search(nums, tempResult, 0);
        return result;
    }
};

在这个版本中,我们使用了一个 set<int>used 来记录在当前递归层已经使用过的数字,从而避免在同一层中重复使用相同的数字。同时,我们在添加元素到 tempResult 之前,检查它是否大于等于 tempResult 的最后一个元素,以确保 tempResult 是非递减的。

另外,我移除了 uniqueResults 集合,因为在这个版本的代码中,我们已经确保了所有生成的子序列都是唯一的,所以不再需要它。

这样修改后的代码应该能够正确地解决问题,且不会生成重复的子序列。

用户:

别的我先不说,你这个set能做到vector的去重?

ChatGPT:

抱歉,我之前的回答中确实有误。std::setstd::unordered_set 不能直接用来存储 vector<int> 类型的元素进行去重,因为默认情况下它们不支持对容器类型的比较。要对 vector<int> 类型的元素进行去重,可以使用 std::set,但必须提供一个比较函数来比较两个 vector<int>。不过,对于这个问题,我们不需要对结果集合中的 vector<int> 进行去重,因为我们可以在搜索过程中避免生成重复的子序列。

在每一层递归中,我们可以使用一个 std::set 来记录这一层已经使用过的元素值,从而避免在同一层递归中使用相同的元素。这是因为我们只关心在同一层递归中是否有重复的元素,而不是跨层递归的元素。

下面是修正后的代码:

#include <vector>
#include <iostream>
using namespace std;

class Solution {
private:
    vector<vector<int>> result;

    void search(vector<int> &nums, vector<int> &tempResult, int start) {
        if (tempResult.size() > 1) {
            result.push_back(tempResult);
        }

        set<int> used; // 用于当前递归层去重
        for (int i = start; i < nums.size(); i++) {
            // 如果当前数字已经在这一层使用过,跳过
            if (used.find(nums[i]) != used.end()) continue;
            // 如果tempResult为空,或者当前数字不小于tempResult中最后一个数字,可以添加
            if (tempResult.empty() || nums[i] >= tempResult.back()) {
                used.insert(nums[i]);
                tempResult.push_back(nums[i]);
                search(nums, tempResult, i + 1);
                tempResult.pop_back();
            }
        }
    }

public:
    vector<vector<int>> findSubsequences(vector<int>& nums) {
        vector<int> tempResult;
        search(nums, tempResult, 0);
        return result;
    }
};

int main() {
    Solution sol;
    vector<int> nums = {4, 6, 7, 7};
    vector<vector<int>> result = sol.findSubsequences(nums);
    for (const auto &subseq : result) {
        for (int num : subseq) {
            cout << num << " ";
        }
        cout << endl;
    }
    return 0;
}

这段代码中,used 集合在每次递归调用时都是一个新的集合,用来记录当前递归层已经使用过的数字,确保在同一层中不会有重复的元素被加入到 tempResult 中。这样,我们在生成结果的过程中就不会产生重复的子序列了。

用户:

终于对了……

ChatGPT:

很高兴听到这个问题终于得到了解决!如果您有其他关于算法、数据结构或编程的问题,或者需要进一步的帮助,请随时告诉我。祝您在算法竞赛中取得优异的成绩!

用户:

当然有。在最初的代码中,我用了if (i > left && nums[i] == nums[i - 1]) ,改进版本中,我们改用了used数组/集合来去重。这么做的区别在哪里?是因为一个要用在有序数组,一个要用在无序数组吗?当然,想法都是一样的:同一层内,如果用了一样的数字(非第一个),那么前面的回溯中必然已经包括了这一结果

ChatGPT:

您的理解是正确的。在递归的回溯问题中,去除重复解的方法通常取决于输入数据是否预先排序。

  1. 当输入数据是有序的,您可以使用类似 if (i > start && nums[i] == nums[i - 1]) 的条件来跳过同一层中已经使用过的元素。因为数组是有序的,所以重复的元素会相邻出现,您只需要检查当前元素是否与前一个元素相同即可。

  2. 当输入数据是无序的,上述方法就不再适用,因为相同的元素可能不是相邻的。这时候,您可以使用一个额外的数据结构(如 used 数组或集合)来记录在当前递归层已经访问过哪些元素。这样做可以保证即使在无序的数组中,同一层递归也不会产生重复的子序列。

在您的问题中,使用 used 数组是为了确保在当前递归层内,相同的数字只被使用一次,即便这个数组没有被排序。这个方法比较通用,因为它不依赖于数组的排序。而且,used 数组的方法在时间复杂度上通常优于排序加上 if (i > start && nums[i] == nums[i - 1]) 的方法,因为排序本身就需要额外的时间(通常是 O(n log n))。

在处理这类问题时,选择哪种去重方法取决于输入数据的特点以及问题的具体要求。如果问题保证了输入数据的有序性,或者排序后不会影响问题的解答,那么第一种方法会更简洁。如果输入数据是无序的,或者排序会改变问题的解答,那么使用 used 数组或集合会更合适。

标签:Non,used,递归,nums,tempResult,vector,Subsequences,result,491
From: https://www.cnblogs.com/Lemon-GPU/p/18145239

相关文章

  • python使用pip安装模块出错 Retrying (Retry(total=0, connect=None, read=None, redi
    问题:在使用python时,使用pip安装psutil出错(pipinstallpsutil),错误信息:Retrying(Retry(total=0,connect=None,read=None,redirect=None,status=None))afterconnectionbrokenby‘SSLError(SSLError(1,u’[SSL:CERTIFICATE_VERIFY_FAILED]certificateverifyfailed......
  • 29天【代码随想录算法训练营34期】第七章 回溯算法part05 (491.递增子序列 * 46.全排
    491.递增子序列如果在最前面加一个uset=set(),这个就是给这一层一个usedset,很好用,不错classSolution:deffindSubsequences(self,nums:List[int])->List[List[int]]:result=[]self.backtracking(nums,[],result,0)returnresult......
  • 在Windows操作系统中,runonce.exe是一个系统进程,用于执行在用户登录时只需执行一次的任
    在Windows操作系统中,runonce.exe是一个系统进程,用于执行在用户登录时只需执行一次的任务或命令。它主要通过注册表中的RunOnce键来实现这些任务的执行。为什么会存在runonce.exe?初始化任务:在用户登录时,有时需要执行一些初始化任务,例如设置默认值、创建用户配置等。这些......
  • Command PhaseScriptExecution failed with a nonzero exit code 错误解决记录
    xCode报这个错误,首先看是哪个文件报错,进入错误日志/Users/fanvil/Library/Developer/Xcode/DerivedData。因为我这边是FBReactNativeSpec这个文件下的错误,所以很容易找到错误日志:/Users/fanvil/Desktop/iOS/Demo/ios/Pods/../../node_modules/react-native/React/FBReactNativeSp......
  • 52 Things: Number 9: What are Shannon's definitions of entropy and information?
    52Things:Number9:WhatareShannon'sdefinitionsofentropyandinformation?52件事:数字9:香农对熵和信息的定义是什么?Thisisthelatestinaseriesofblogpoststoaddressthelistof'52ThingsEveryPhDStudentShouldKnowToDoCryptography':a......
  • Python中global和nonlocal关键字的妙用:变量管理技巧
        概要在Python中编写函数时,经常会遇到需要在函数内部访问和修改外部变量的情况。在这种情况下,我们可以使用 global 和 nonlocal 关键字来声明变量的作用域,以便正确地访问和修改这些变量。本文将深入探讨 global 和 nonlocal 的用法,包括详细的示例代码和......
  • SELECT list is not in GROUP BY clause and contains nonaggregated column 'uav.cas
     mysql5.7以上版本抛出错误,SELECTlistisnotinGROUPBYclauseandcontainsnonaggregatedcolumn'uav.case_board.port'whichisnotfunctionallydependentoncolumnsinGROUPBYclause;thisisincompatiblewithsql_mode=only_full_group_bygrougby在5......
  • TurnOnLamps
    贪心贪心的从下往上,对于当前位,如果这一位是任意的,那么如果它有儿子可以向上,那么当前点选择为翻转是肯定不劣的,而如果没有儿子向上,那么当前点选中一定是不优的然后数向上的儿子个数,每\(2\)个可以合并为\(1\)条路径如果一个点合并后不满足限制,则通过从当前点重新出发一条链来......
  • Canon CMOS图像传感器应用和选型
    一、CCD和CMOS图像传感器常见的数字感光元件有两种:CCD(电荷耦合器件)和CMOS(互补金属氧化物半导体)。1980年代,CCD进入消费级市场,并长期占据中高端市场。CMOS图像传感器最初作为廉价、低画质的代名词,但是在2008年之后逐渐成为市场主流。如今已经广泛应用于手机、汽车、安防、医疗、......
  • CF1924D Balanced Subsequences 题解
    先判掉\(k>\min(n,m)\)的情况。首先有个明显的计算最长合法括号子序列的贪心方法:初始一个值为\(0\)的计数器,从左到右枚举每个字符,遇到(时将计数器加一,遇到)时如果计数器不为\(0\)就减一然后将答案加一。考虑绘制它的折线图。初始时纵坐标为\(0\),从左到右枚举每个字符......