Leetcode0908. 最小差值 I(simple)

1. 问题描述

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

在一个操作中,您可以选择 0 <= i < nums.length 的任何索引 i 。将 nums[i] 改为 nums[i] + x ,其中 x 是一个范围为 [-k, k] 的整数。对于每个索引 i ,最多 只能 应用 一次 此操作。

nums 的 分数 是 nums 中最大和最小元素的差值。 

在对  nums 中的每个索引最多应用一次上述操作后,返回 nums 的最低 分数 。

示例 1:

输入:nums = [1], k = 0
输出:0
解释:分数是 max(nums) - min(nums) = 1 - 1 = 0。

示例 2:

输入:nums = [0,10], k = 2
输出:6
解释:将 nums 改为 [2,8]。分数是 max(nums) - min(nums) = 8 - 2 = 6。

示例 3:

输入:nums = [1,3,6], k = 3
输出:0
解释:将 nums 改为 [4,4,4]。分数是 max(nums) - min(nums) = 4 - 4 = 0。

提示:

  • 1 <= nums.length <= 10^4
  • 0 <= nums[i] <= 10^4
  • 0 <= k <= 10^4

2. 思路与算法

        每个数都有变还是不变的选择,这就是2^n种不同的选择。进一步调节的范围在[-k,k]之间可以任意选择,所以可能生成的序列数为(2k+1)2^n。暴力枚举显然不是明智的选择。

        每个数选择变或是不变都是为了同一个目的:缩小数据的范围。

        直观地来说应该是较小的数往大调,较大的数往小调有助于缩小数据的范围。

        由于可以在[-k,k]范围内任意选择,所以事实上只需要考虑最大数与最小数的调节即可。

 

3. 代码       

class Solution:
    def smallestRangeI(self, nums: List[int], k: int) -> int:
        return max(0,(max(nums)-k) - (min(nums)+k))

        

        执行用时:40 ms, 在所有 Python3 提交中击败了70.05%的用户

        内存消耗:15.9 MB, 在所有 Python3 提交中击败了36.23%的用户

        

        老实讲,在最后想通了要点之前,懵逼了很久。。心想这能是simple?但是,想通要点并写出代码很容易,要给出严格的证明却不是一件容易的事情(这种基于数学推断的方式呃解决方案需要严格的最优性证明作为支撑)。以下为力扣官解给出的证明供感兴趣者参考:

        另外参见:Leetcode0910. 最小差值 II(medium)

        910的约束为对于给定的k只能做+k或者-k的调节,第一感或许会觉得可选择范围更小了,那不是更简单嘛。其实不然,本题与910的对比类似于线性规划与整数规划问题的对比。整数规划是线性规划的一个子类,但是比线性规划难,因为选择范围更小从另一个角度来理解就意味着约束条件更紧,这就导致找到满足约束条件的最优解更难。

        回到总目录:笨牛慢耕的Leetcode每日一题总目录(动态更新。。。)

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

笨牛慢耕

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值