【练习】【类似于子集问题】力扣491. 非递减子序列/递增子序列

news/2025/2/24 8:21:24

题目

  1. 非递减子序列

给你一个整数数组 nums ,找出并返回所有该数组中不同的递增子序列,递增子序列中 至少有两个元素 。你可以按 任意顺序 返回答案。

数组中可能含有重复元素,如出现两个整数相等,也可以视作递增序列的一种特殊情况。

示例 1:

输入:nums = [4,6,7,7]

输出:[[4,6],[4,6,7],[4,6,7,7],[4,7],[4,7,7],[6,7],[6,7,7],[7,7]]

示例 2:

输入:nums = [4,4,3,2,1]

输出:[[4,4]]

来源:力扣491. 非递减子序列


思路(注意事项)

  1. 用数组记录同一层是否有重复的元素
  • !path.empty()才可以保证有path.back()
  • 至于为什么回溯的时候没有将标记重置回0,是因为要求同一层不能有重复的元素,如果重置0,则后续元素判断时,会忘记之前是否有与其相等的元素。
  1. 用哈希表记录同一层是否有重复的元素

纯代码1

class Solution {
private:
    vector<vector<int>> ans;
    vector<int> path;

    void backtracking (vector<int>& nums, int start)
    {
        if (path.size() >= 2) ans.push_back(path);
        
        vector<int> tmp(201, 0); // [-100,100]
        for (int i = start; i < nums.size(); i ++)
        {
            if (!path.empty() && nums[i] < path.back() || tmp[100 + nums[i]] == 1) continue;
            tmp[100 + nums[i]] = 1;
            path.push_back(nums[i]);
            backtracking(nums, i + 1);
            path.pop_back();
        }
    }
public:
    vector<vector<int>> findSubsequences(vector<int>& nums) {
        backtracking (nums, 0);
        return ans;
    }
};

题解1(加注释)

class Solution {
private:
    vector<vector<int>> ans;  // 存储所有符合条件的子序列
    vector<int> path;         // 存储当前递归路径中的子序列

    // 回溯函数,用于生成所有非递减子序列
    void backtracking(vector<int>& nums, int start) {
        // 如果当前路径中的子序列长度大于等于 2,将其加入结果
        if (path.size() >= 2) ans.push_back(path);

        // 使用临时数组 tmp 去重,确保同一层级中不会重复选择相同的元素
        vector<int> tmp(201, 0);  // 数组大小为 201,覆盖范围 [-100, 100]

        // 遍历数组,从 start 开始
        for (int i = start; i < nums.size(); i++) {
            // 如果 path 不为空且当前元素小于 path 的最后一个元素,跳过(确保子序列非递减)
            // 或者当前元素已经在本层级使用过,跳过(去重)
            if (!path.empty() && nums[i] < path.back() || tmp[100 + nums[i]] == 1) continue;

            // 标记当前元素为已使用
            tmp[100 + nums[i]] = 1;

            // 将当前元素加入路径
            path.push_back(nums[i]);

            // 递归调用,处理下一个元素
            backtracking(nums, i + 1);

            // 回溯:移除当前元素,尝试其他可能性
            path.pop_back();
        }
    }

public:
    // 主函数,生成所有非递减子序列
    vector<vector<int>> findSubsequences(vector<int>& nums) {
        // 从索引 0 开始回溯
        backtracking(nums, 0);

        // 返回所有符合条件的子序列
        return ans;
    }
};

纯代码2

class Solution {
private:
    vector<vector<int>> ans;
    vector<int> path;

    void backtracking (vector<int>& nums, int start)
    {
        if (path.size() >= 2) ans.push_back(path);

        unordered_set<int> st;
        for (int i = start; i < nums.size(); i ++)
        {
            if (!path.empty() && nums[i] < path.back() || st.find (nums[i]) != st.end()) continue;
            st.insert (nums[i]);
            path.push_back(nums[i]);
            backtracking(nums, i + 1);
            path.pop_back();
        }
    }
public:
    vector<vector<int>> findSubsequences(vector<int>& nums) {
        backtracking (nums, 0);
        return ans;
    }
};

题解2(加注释)

class Solution {
private:
    vector<vector<int>> ans;  // 存储所有符合条件的子序列
    vector<int> path;         // 存储当前递归路径中的子序列

    // 回溯函数,用于生成所有非递减子序列
    void backtracking(vector<int>& nums, int start) {
        // 如果当前路径中的子序列长度大于等于 2,将其加入结果
        if (path.size() >= 2) ans.push_back(path);

        // 使用哈希集合 st 去重,确保同一层级中不会重复选择相同的元素
        unordered_set<int> st;

        // 遍历数组,从 start 开始
        for (int i = start; i < nums.size(); i++) {
            // 如果 path 不为空且当前元素小于 path 的最后一个元素,跳过(确保子序列非递减)
            // 或者当前元素已经在本层级使用过,跳过(去重)
            if (!path.empty() && nums[i] < path.back() || st.find(nums[i]) != st.end()) continue;

            // 将当前元素加入哈希集合,标记为已使用
            st.insert(nums[i]);

            // 将当前元素加入路径
            path.push_back(nums[i]);

            // 递归调用,处理下一个元素
            backtracking(nums, i + 1);

            // 回溯:移除当前元素,尝试其他可能性
            path.pop_back();
        }
    }

public:
    // 主函数,生成所有非递减子序列
    vector<vector<int>> findSubsequences(vector<int>& nums) {
        // 从索引 0 开始回溯
        backtracking(nums, 0);

        // 返回所有符合条件的子序列
        return ans;
    }
};

http://www.niftyadmin.cn/n/5864114.html

相关文章

WPF框架学习

WPF 可以想winfrom 那样在cs文件修改 属性数据&#xff1b; 为了前后端分离 而解耦合&#xff0c;有了M-V-VM模式 常见框架有 MVVMlight / Prism 等 ------------------------------------------------------------------------------------- 一、前提&#xff1a;有一定基…

Xcode如何高效的一键重命名某个关键字

1.选中某个需要修改的关键字&#xff1b; 2.右击&#xff0c;选择Refactor->Rename… 然后就会出现如下界面&#xff1a; 此时就可以一键重命名了。 还可以设置快捷键。 1.打开Settings 2.找到Key Bindings 3.搜索rename 4.出现三个&#xff0c;点击一个地方设置后其…

财务运营域——营收稽核系统设计

摘要 本文主要介绍了营收稽核系统的背景、特点与作用。营收稽核系统的产生源于营收管理复杂性、财务合规与审计需求、提升数据透明度与决策效率、防范舞弊与风险管理、技术进步与自动化需求、多元化业务模式以及跨部门协作与数据整合等多方面因素。其特点包括自动化与智能化、…

文章精读篇——用于遥感小样本语义分割的可学习Prompt

题目&#xff1a;Learnable Prompt for Few-Shot Semantic Segmentation in Remote Sensing Domain 会议&#xff1a;CVPR 2024 Workshop 论文&#xff1a;10.48550/arXiv.2404.10307 相关竞赛&#xff1a;https://codalab.lisn.upsaclay.fr/competitions/17568 年份&#…

华为 网络安全 认证

&#x1f345; 点击文末小卡片 &#xff0c;免费获取网络安全全套资料&#xff0c;资料在手&#xff0c;涨薪更快 华为 网络安全 认证&#xff1a;保障信息安全的重要一环 在数字化时代的今天&#xff0c;网络安全成为了企业和个人都需要高度重视的问题。尤其是在企业信息化的…

langchain系列(四)- LangChain 的RAG原理与代码实现

导读 环境&#xff1a;OpenEuler、Windows 11、WSL 2、Python 3.12.3 langchain 0.3 背景&#xff1a;前期忙碌的开发阶段结束&#xff0c;需要沉淀自己的应用知识&#xff0c;过一遍LangChain 时间&#xff1a;20250223 说明&#xff1a;技术梳理&#xff0c;使用LangChain…

【漫话机器学习系列】101.特征选择法之Lasso(Lasso For Feature Selection)

Lasso 特征选择法详解 1. Lasso 回归简介 Lasso&#xff08;Least Absolute Shrinkage and Selection Operator&#xff0c;最小绝对收缩和选择算子&#xff09;是一种基于 L1 范数正则化的线性回归方法。它不仅能够提高模型的泛化能力&#xff0c;还可以自动进行特征选择&am…

用AI写游戏3——deepseek实现kotlin android studio greedy snake game 贪吃蛇游戏

项目下载 https://download.csdn.net/download/AnalogElectronic/90421306 项目结构 就是通过android studio 建空项目&#xff0c;改下MainActivity.kt的内容就完事了 ctrlshiftalts 看项目结构如下 核心代码 MainActivity.kt package com.example.snakegame1// MainA…