当前位置: 首页 > article >正文

20240209-最大可整分子集

题目要求

给定一组不同的正整数 nums,返回最大的子集 answer,使得该子集中的每一对元素(answer[i], answer[j])都满足:

answer[i] % answer[j] == 0,或
answer[j] % answer[i] == 0
如果有多个解决方案,则返回其中任意一个。

Example 1:

Input: nums = [1,2,3]
Output: [1,2]
Explanation: [1,3] is also accepted.

Example 2:

Input: nums = [1,2,4,8]
Output: [1,2,4,8]

思路

再次推荐一下labuladong,东哥的题解和思路写的都很好。

首先通过观察发现这个题目是一个穷举问题,但是由于数据量巨大,所以阶乘级别的时间复杂度是不行的。

再思考一下,在数组中寻找一个特定要求的子序列,那么如果我们把数组进行排列,这个题目就变成了寻找“最长上升子序列”类似的题目,因为我们要求每两个数之间具备整除关系,这就意味着其中一个一定大于等于另一个。

排序很重要,假设一个符合题目要求的子集 [1,2,4],那么由于我排了序,如果想扩充这个集合,接下来我只要到一个能够整除 4 的数就行了,比如 8。我不用考虑前面的数字了,因为只要能整除最大的 4 就一定能整除别的。

这样一来,题目就变成了:寻找一个最长递增子序列,且要求每个元素都能够整除前面那个元素

那么我们只需要定义一个dp数组,dp[i]代表了以nums[i]结尾的最长上升可整除子序列的个数。要求新加入的元素能够整除前一个元素,所以这里在遍历i的同时,可能还需要遍历之前所有的数来检查是否能够整除:dp[i] = max(dp[i], dp[j]+1) {if nums[i] % nums[j] == 0}

现在我们还有一个点需要注意,因为我们需要返回的是最长的子集,因此我们不能只存储最长的子集。我们需要在判断dp[i]和dp[j]的大小时,动态的存储i对应的前一位的位置j。

代码

注意在回溯获得最长子集的过程中采用的for循环。

class Solution {
public:
    vector<int> largestDivisibleSubset(vector<int>& nums) {
        sort(nums.begin(), nums.end());
        int n = nums.size();
        vector<int> dp(n, 1), parent(n, -1);
        vector<int> result;
        int maxIndex = 0;
        for (int i = 1; i < n; ++i) {
            for (int j = 0; j < i; ++j) {
                if (nums[i] % nums[j] == 0 && dp[i] < dp[j] + 1) {
                    dp[i] = dp[j] + 1;
                    parent[i] = j;
                    if (dp[i] > dp[maxIndex]) {
                        maxIndex = i;
                    }
                }
            }
        }
        for (int i = maxIndex; i != -1; i = parent[i]) {
            result.push_back(nums[i]);
        }
        return result;
    }
};

时空复杂度

时间复杂度O(N^2),其中 N 是 nums 中元素的数量,因为存在双层循环来找到最大的可整除子集。排序的时间复杂度是O(NlogN)

空间复杂度:O(N),用于存储 dpparent 数组,以及构建结果子集。


http://www.kler.cn/a/233044.html

相关文章:

  • 【ROS2】数据记录(ros2 bag)详解
  • 我国无人机新增实名登记110.3 万架,累计完成飞行2666万小时
  • 鸿蒙面试 2025-01-10
  • 【ArcGIS初学】产生随机点计算混淆矩阵
  • 主数据系统建设模式分析
  • Microsoft
  • c#: 表达式树的简化
  • 科研绘图-半小提琴图-
  • List 差集
  • 【开源】JAVA+Vue+SpringBoot实现房屋出售出租系统
  • vue父子组件通讯的几种方式总结学习
  • 基于SpringBoot的记账系统项目
  • Vagrant 虚拟机工具基本操作指南
  • echarts 一条折线图上显示不同颜色
  • 【Android】GridLayout实现等比布局
  • DC-9靶机渗透详细流程
  • 每日五道java面试题之java基础篇(一)
  • 飞桨自然语言处理框架 paddlenlp的 trainer
  • openssl3.2 - exp - buffer to BIO
  • HarmonyOS SDK 助力新浪新闻打造精致易用的新闻应用
  • 【ETOJ P1046】斐波那契数列 题解(数学+动态规划)
  • Electron+Vue实现仿网易云音乐实战
  • python 基础知识点(蓝桥杯python科目个人复习计划35)
  • 【开源】JAVA+Vue+SpringBoot实现实验室耗材管理系统
  • 前端图片转base64 方法
  • MinGW/MSYS/GCC/GNU/MSVC/Clang/LLVM都是什么