zoukankan      html  css  js  c++  java
  • 330. Patching Array

    Given a sorted positive integer array nums and an integer n, add/patch elements to the array such that any number in range [1, n] inclusive can be formed by the sum of some elements in the array. Return the minimum number of patches required.

    Example 1:

    Input: nums = [1,3], n = 6
    Output: 1 
    Explanation:
    Combinations of nums are [1], [3], [1,3], which form possible sums of: 1, 3, 4.
    Now if we add/patch 2 to nums, the combinations are: [1], [2], [3], [1,3], [2,3], [1,2,3].
    Possible sums are 1, 2, 3, 4, 5, 6, which now covers the range [1, 6].
    So we only need 1 patch.

    Example 2:

    Input: nums = [1,5,10], n = 20
    Output: 2
    Explanation: The two patches can be [2, 4].
    

    Example 3:

    Input: nums = [1,2,2], n = 5
    Output: 0
    class Solution {
        public int minPatches(int[] nums, int n) {
            int patchCount = 0;
            int index = 0;
            long miss = 1;
            while (miss <= n) {
                if (index < nums.length && nums[index] <= miss) {
                    miss += nums[index++];
                } else {
                    miss += miss;
                    patchCount++;
                }
            }
            return patchCount;
    }}

    看起来是要补最小可以用的数字,大的不行?比如【1,5,10】,20就可以补9?实际上只能补2,3.??

    ??

    实际上不行,题目是要【1,n】之间所有数字能被任意子集之和cover,如果补9,就不能帮我们cover比如7,8。

    意思就是subset和必可以成为【1,n】的任意一项。

    首先可以确定的是,

    • nums中必然包含1,如果不包含1,那么[1,n]这个范围中的1就没法实现
    • 其次数组中的元素不能重复使用,如果允许重复使用,那么把1重复多次,就可以组成任意整数。

    miss[0,n]中缺少的最小整数,意味着我们可以实现[0,miss)范围内的任意整数。

    1. 如果数组中有某个整数x<=miss, 那么我们可以把[0,miss)区间的所有整数加上x,区间变成了[x, miss+x),由于区间[0,miss)[x, miss+x)重叠,两个区间可以无缝连接起来,意味着我们可以把区间[0,miss)扩展到[0, miss+x)
    2. 如果数组中不存在小于或等于miss的元素,则区间[0,miss)[x, miss+x) 脱节了,连不起来。此时我们需要添加一个数,最大限度的扩展区间[0, miss)。那添加哪个数呢?当然是添加miss本身,这样区间[0,miss)[miss, miss+miss)恰好可以无缝拼接。

    举个例子,令nums=[1, 2, 4, 13, 43]n=100,我们需要让[1,100]内的数都能够组合出来。

    使用数字1,2,4,我们可以组合出[0, 8)内的所有数,但无法组合出8,由于下一个数是13,比8大,根据规则2,我们添加8,把区间从[0,8)扩展到[0,16)

    下一个数是13,比16小,根据规则1,我们可以把区间从[0,16)扩展到[0,29)

    下一个数是43,比29大,根据规则2,添加29,把区间从[0,29)扩大到[0,58)

    由于43比58小,根据规则1,可以把区间从[0,58)扩展到[0,101),刚好覆盖了[1,100]内的所有数。

    +

    最终结果是添加2个数,8和29,就可以组合出[1,100]内的所有整数。


    ————————————————
    版权声明:本文为CSDN博主「Code_Granker」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
    原文链接:https://blog.csdn.net/happyaaaaaaaaaaa/article/details/50899547

  • 相关阅读:
    linux 上安裝lnmp
    html 禁用点击事件
    nftables 是什么? 提供什么功能? 如何使用?
    ESXi主机RAID卡_HBA卡_网卡 型号_固件_驱动查询
    Celery Beat定时任务
    Centos 7/8 安装Rabbit-MQ
    Celery 最佳实践
    Django 3.0 + Celery 4.4 + RabbitMQ
    C语言Socket示例
    深入理解计算机系统 — 读书笔记
  • 原文地址:https://www.cnblogs.com/wentiliangkaihua/p/11595476.html
Copyright © 2011-2022 走看看