LeetCode 3634. 使数组平衡的最少移除数目【排序+滑动窗口】1453

本文属于「征服LeetCode」系列文章之一,这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁,本系列将至少持续到刷完所有无锁题之日为止;由于LeetCode还在不断地创建新题,本系列的终止日期可能是永远。在这一系列刷题文章中,我不仅会讲解多种解题思路及其优化,还会用多种编程语言实现题解,涉及到通用解法时更将归纳总结出相应的算法模板。

为了方便在PC上运行调试、分享代码文件,我还建立了相关的仓库:https://github.com/memcpy0/LeetCode-Conquest。在这一仓库中,你不仅可以看到LeetCode原题链接、题解代码、题解文章链接、同类题目归纳、通用解法总结等,还可以看到原题出现频率和相关企业等重要信息。如果有其他优选题解,还可以一同分享给他人。

由于本系列文章的内容随时可能发生更新变动,欢迎关注和收藏征服LeetCode系列文章目录一文以作备忘。

给你一个整数数组 nums 和一个整数 k

如果一个数组的 最大 元素的值 至多 是其 最小 元素的 k 倍,则该数组被称为是 平衡 的。

你可以从 nums 中移除 任意 数量的元素,但不能使其变为 数组。

返回为了使剩余数组平衡,需要移除的元素的 最小 数量。

注意:大小为 1 的数组被认为是平衡的,因为其最大值和最小值相等,且条件总是成立。

示例 1:

java 复制代码
输入:nums = [2,1,5], k = 2
输出:1

解释:

  • 移除 nums[2] = 5 得到 nums = [2, 1]
  • 现在 max = 2, min = 1,且 max <= min * k,因为 2 <= 1 * 2。因此,答案是 1。

示例 2:

java 复制代码
输入:nums = [1,6,2,9], k = 3
输出:2

解释:

  • 移除 nums[0] = 1nums[3] = 9 得到 nums = [6, 2]
  • 现在 max = 6, min = 2,且 max <= min * k,因为 6 <= 2 * 3。因此,答案是 2。

示例 3:

java 复制代码
输入:nums = [4,6], k = 2
输出:0

解释:

  • 由于 nums 已经平衡,因为 6 <= 4 * 2,所以不需要移除任何元素。

提示:

  • 1 <= nums.length <= 10^5
  • 1 <= nums[i] <= 10^9
  • 1 <= k <= 10^5

解法 排序+滑动窗口

由于我们只关心剩余元素的最小最大值,不关心元素的顺序,所以可从小到大排序,方便后续计算。

排序后,要想整个数组平衡,我们必须让数组左右两端的最大最小值满足题意,不满足的话,需要去掉最大值或最小值。到底是去掉最大值还是最小值呢?我们还要满足移除最少元素个数剩余最大元素个数)的题意。

此时本题和[[LeetCode 1423. 可获得的最大点数【定长滑窗,逆向和正向思维】1574]]十分类似,1423题也只能从开头或末尾拿牌,但该题能拿走的牌数是固定的,本题能移除的元素个数则是不定的。为了移除最少元素,我们要计算排序后平衡子数组的最大长度

排序后,枚举最大值 m x = n u m s [ i ] mx = nums[i] mx=nums[i] ,那最小值为 m n = n u m s [ l e f t ] mn = nums[left] mn=nums[left] 必须满足 m n × k ≥ m x mn \times k \ge mx mn×k≥mx在 [ m n , m x ] [mn, mx] [mn,mx] 中的元素保留,其余元素丢掉。由于排序了,所以这些元素在数组中是连续 的,问题转换为一个标准的滑动窗口 模型。如果不满足上式,就把 l e f t left left 加一,直到满足上式。

内层循环结束后,用窗口长度 i − l e f t + 1 i - left + 1 i−left+1 更新保留元素个数的最大值 m a x S a v e maxSave maxSave 。

java 复制代码
class Solution {
    public int minRemoval(int[] nums, int k) {
        Arrays.sort(nums);
        int maxSave = 0;
        int left = 0;
        for (int i = 0; i < nums.length; i++) {
            while ((long) nums[left] * k < nums[i]) {
                left++;
            }
            maxSave = Math.max(maxSave, i - left + 1);
        }
        return nums.length - maxSave;
    }
}
cpp 复制代码
class Solution {
public:
    int minRemoval(vector<int>& nums, int k) {
        ranges::sort(nums);
        int max_save = 0, left = 0;
        for (int i = 0; i < nums.size(); i++) {
            while (1LL * nums[left] * k < nums[i]) {
                left++;
            }
            max_save = max(max_save, i - left + 1);
        }
        return nums.size() - max_save;
    }
};
python 复制代码
class Solution:
    def minRemoval(self, nums: List[int], k: int) -> int:
        nums.sort()
        max_save = left = 0
        for i, mx in enumerate(nums):
            while nums[left] * k < mx:
                left += 1
            max_save = max(max_save, i - left + 1)
        return len(nums) - max_save
go 复制代码
func minRemoval(nums []int, k int) int {
	slices.Sort(nums)
	maxSave, left := 0, 0
	for i, mx := range nums {
		for nums[left]*k < mx {
			left++
		}
		maxSave = max(maxSave, i-left+1)
	}
	return len(nums) - maxSave
}
rust 复制代码
impl Solution {
    pub fn min_removal(mut nums: Vec<i32>, k: i32) -> i32 {
        nums.sort_unstable();
        let mut max_save = 0;
        let mut left = 0;
        for (i, &x) in nums.iter().enumerate() {
            while (nums[left] as i64) * (k as i64) < x as i64 {
                left += 1;
            }
            max_save = max_save.max(i - left + 1);
        }
        (nums.len() - max_save) as _
    }
}
js 复制代码
var minRemoval = function(nums, k) {
    nums.sort((a, b) => a - b);
    const n = nums.length;
    let maxSave = 0;
    let left = 0;
    for (let i = 0; i < n; i++) {
        while (nums[left] * k < nums[i]) {
            left++;
        }
        maxSave = Math.max(maxSave, i - left + 1);
    }
    return n - maxSave;
};
c 复制代码
#define MAX(a, b) ((b) > (a) ? (b) : (a))

int cmp(const void* a, const void* b) {
    return *(int*)a - *(int*)b;
}

int minRemoval(int* nums, int numsSize, int k) {
    qsort(nums, numsSize, sizeof(int), cmp);
    int max_save = 0;
    int left = 0;
    for (int i = 0; i < numsSize; i++) {
        while (1LL * nums[left] * k < nums[i]) {
            left++;
        }
        max_save = MAX(max_save, i - left + 1);
    }
    return numsSize - max_save;
}

复杂度分析:

  • 时间复杂度: O ( n log ⁡ n ) O(n\log n) O(nlogn) ,其中 n n n 是 n u m s nums nums 的长度。瓶颈在排序上。
  • 空间复杂度: O ( 1 ) O(1) O(1) 。忽略排序的栈开销。
相关推荐
python_DONG19 小时前
响应面法(Response Surface Methodology, RSM)单目标优化算法
算法·数学建模
6Hzlia20 小时前
【Hot 100 刷题计划】 LeetCode 108. 将有序数组转换为二叉搜索树 | C++ 分治法详解
c++·算法·leetcode
itzixiao20 小时前
L1-051 打折(5分)[java][python]
java·python·算法
贾斯汀玛尔斯21 小时前
每天学一个算法--Aho–Corasick 自动机
java·linux·算法
re林檎21 小时前
八大排序算法(C++实现)
c++·算法·排序算法
淘气包海鸟21 小时前
雷达度量衡量
人工智能·算法·机器学习·信息与通信
睡觉就不困鸭21 小时前
第12天 多数元素
算法·哈希算法·散列表
cpp_250121 小时前
P2639 [USACO09OCT] Bessie‘s Weight Problem G
数据结构·算法·动态规划·题解·洛谷·背包dp
郝学胜-神的一滴21 小时前
[力扣 227] 双栈妙解表达式计算:从思维逻辑到C++实战,吃透反向波兰式底层原理
java·前端·数据结构·c++·算法
LDG_AGI21 小时前
【搜索引擎】Elasticsearch(六):向量搜索深度解析:从参数原理到混合查询实战
人工智能·深度学习·算法·elasticsearch·机器学习·搜索引擎