首页 > 其他分享 >2835. 使子序列的和等于目标的最少操作次数-360

2835. 使子序列的和等于目标的最少操作次数-360

时间:2023-08-28 20:57:13浏览次数:37  
标签:target nums int 构造 使子 long 数组 2835 360

使子序列的和等于目标的最少操作次数

给你一个下标从 0 开始的数组 nums ,它包含 非负 整数,且全部为 2 的幂,同时给你一个整数 target 。

一次操作中,你必须对数组做以下修改:

选择数组中一个元素 nums[i] ,满足 nums[i] > 1 。
将 nums[i] 从数组中删除。
在 nums 的 末尾 添加 两个 数,值都为 nums[i] / 2 。
你的目标是让 nums 的一个 子序列 的元素和等于 target ,请你返回达成这一目标的 最少操作次数 。如果无法得到这样的子序列,请你返回 -1 。

数组中一个 子序列 是通过删除原数组中一些元素,并且不改变剩余元素顺序得到的剩余数组。

示例 1:

输入:nums = [1,2,8], target = 7
输出:1
解释:第一次操作中,我们选择元素 nums[2] 。数组变为 nums = [1,2,4,4] 。
这时候,nums 包含子序列 [1,2,4] ,和为 7 。
无法通过更少的操作得到和为 7 的子序列。
示例 2:

输入:nums = [1,32,1,2], target = 12
输出:2
解释:第一次操作中,我们选择元素 nums[1] 。数组变为 nums = [1,1,2,16,16] 。
第二次操作中,我们选择元素 nums[3] 。数组变为 nums = [1,1,2,16,8,8] 。
这时候,nums 包含子序列 [1,1,2,8] ,和为 12 。
无法通过更少的操作得到和为 12 的子序列。
示例 3:

输入:nums = [1,32,1], target = 35
输出:-1
解释:无法得到和为 35 的子序列。

提示:

1 <= nums.length <= 1000
\(1 <= nums[i] <= 2^{30}\)
nums 只包含非负整数,且均为 2 的幂。
\(1 <= target < 2^{31}\)

解题思路

见代码注释。
c++由于是强类型语言,所以需要注意数据溢出的问题,比如int和long long int,要是以后实在找不到哪里溢出还是直接python吧orz。学到了一个c++的新语法:1LL,可以告诉编译器将1解释为long long int。

code

class Solution:
    '''
    1. 什么情况下结果不存在
    sum(nums) < target时结果不存在;
    如果sum(nums) >= target,那么最糟糕的情况是全部拆分为1,也是可以有解的

    2.通过二进制的方式思考(2的幂)

    3.从低位到高位思考
    要求的是最小的操作次数,那么最好的情况就是可以使用现有的数组元素构造出target目标元素,退而求其次就是需要使用 
    数组元素构造出target中的一部分,也就是需要从低到高进行构建

    数组中的所有小于等于2^i的元素和>=2^i,能否不操作构造出2^i
    1.可以的,通过数学归纳法可以证明
        2^0=1 -> s>=1,那么必然是有1的,可以直接构造得出
        2^1=2 -> s>=2,如果有2,直接可以构造,如果没有2,那么至少有两个1,可以构造
        2^2=4 -> s>=4,如果有4,直接可以构造,如果没有4,那么数组中的元素都是<=2的,通过第二点可知,可以构造出一
        个2,同时由于s>=4,那么剩余的结果>=2,根据2还是可以构造出一个,最后构造出4
        ……
        2^k   -> s >= 2^k可以构造出
        2^k+1 -> 2 >= 2^k+1,如果有2^k+1,那么可以直接构造出结果,如果不存在,那么数组中的元素都是<=2^k的,通过上
        一点可知可以构造出一个2^k,在构造出一个2^k后,剩余的结果是>=2^k的,还可以构造出一个2^k,最后构造出结果
    2. 算法过程
        遍历target的每一个二进制位,1则尝试构造,0则不用考虑
        1. 计算s,如果s >= 2^i,可以直接构造,从s中减去2^i
        2. 如果s<2^i,需要从比2^i大的数2^j中进行拆分
            拆分可以得到2^j-1,2^j-2.....2^i+1,2^i,2^i
            可以看到:
            1.操作次数为j - i
            2.不仅得到i,还有j-1,j-2,....,i+1,也就是下一位可以从j开始
            3.这里想不明白的因为涉及到s的加减,可以直接构造的话要减去,不可以直接构造的话得拆分,拆分后s应当如何 
            计算如果从j开始的话
            4.所以还是朴素写法吧,不要直接跳,有点想不明白怎么写       
    '''
    def minOperations(self, nums: List[int], target: int) -> int:
        
        if(sum(nums) < target):
            return -1

        cnt = Counter(nums)
        ans = s = i = 0
        flag = 0
        while((1 << i) <= target):
            
            s += cnt[1<<i] * (1 << i)

            if((target>>i) & 1):
                
                if(s >= (1<<i)):
                    s -= (1<<i)
                    i += 1
                    continue
                
                #查找比1<<i大的数
                j = i + 1
                while(cnt[1<<j] == 0):
                    j += 1
                
                ans += j - i
                for k in range(i,j):
                    cnt[1<<k] += 1
                cnt[1<<j] -= 1
                s += 1 <<i
                i += 1
                
            else:
                i += 1

        return ans
class Solution {
public:
    int minOperations(vector<int>& nums, int target) {
        long long int sum = 0;
        
        for(auto item : nums) sum += item;

        if(sum < target) return -1;

        unordered_map<int,int> cnt;

        for(auto item : nums) cnt[item]++;

        int ans = 0,i = 0;
        long long int s = 0;

        while((1LL<<i) <= target)
        {
            cout<<i<<endl;
            s += (long long int)cnt[1<<i] * (long long int)(1<<i);
            if((target>>i) & 1)
            {
                
                if(s >= (1<<i))
                {
                    s -= 1 << i;
                    i ++;    
                    continue;
                }

                int j = i + 1;
                //cout<<(1<<i)<<" "<<j<<endl;
                while(cnt.find(1<<j) == cnt.end()) 
                { 
                    //cout<<"in?"<<endl;
                    j++;
                }
                
                ans += j - i;

                for(int k = i;k < j;k++) cnt[1<<k] ++;
                cnt[1 << j] --;
                s += 1 << i;
                i ++;
                //cout<<i<<(1<<i)<<endl;

            }
            else i ++;
            
        }

        return ans;
    }
};

标签:target,nums,int,构造,使子,long,数组,2835,360
From: https://www.cnblogs.com/huangxk23/p/17663370.html

相关文章

  • Asrock-Z690-PG-Reptide i5-13600kf电脑 Hackintosh 黑苹果引导文件
    硬件配置(需要下载请百度搜索:黑果魏叔)硬件型号驱动情况主板AsrockZ690PGReptide处理器i5-13600kfRaptorLake(Undervolted)已驱动内存2x16GbDDR43600ADATAXPG已驱动硬盘1TbNetacNV7000NVMEM2(PCI-e4.0)已驱动显卡RadeonRX6600PowerColorFighter8Gb已驱动声卡瑞昱......
  • vue项目在360浏览器兼容模式下SCRIPT1002: 语法错误以及“fetch”未定义问题解决
    使用360浏览器的兼容模式,vue项目页面空白,打开控制台,发现如下报错:SCRIPT1002:语法错误 解决方法如下:1、安装依赖npminstall--savecore-jsregenerator-runtime2、在main.js引入import'core-js/stable';import'regenerator-runtime/runtime';3、在babel.confi......
  • 360安全卫士如何关闭精选弹窗
    360安全卫士如何关闭精选弹窗https://www.comcw.cn/jc/11287.html......
  • 360浏览器上的对js支持的一些特别的问题
    在做web项目时候,基本测试了IE,火狐,chrome,基本就觉得差不多了。但是国内的用户,经常反而使用搜狗,360之类的伪浏览器,因为他们都稀里糊涂的被捆绑了。这里要说的是,在其他浏览器很兼容的东西,但是到360这边就出问题了。2个表格显示的例子。例1:jq('#classManageTable').append(data);这个......
  • 第三代网关,POE级联蓝牙网关VDB3601,至多可连接38台蓝牙设备
    第三代蓝牙网关,网关集成了蓝牙4.2/5.0+WiFi无线协议,采用双网口设计,1台主蓝牙网关可级联多个从蓝牙网关设备,至多支持远距离连接和控制38台蓝牙设备的蓝牙网关VDB3601,支持双蓝牙模组、485通信、可兼容4G/Cat.1模块,安装更方便,能适用无法布线的特殊场景。VDB3601升级了POE的兼容性和可......
  • 龙蜥社区安全联盟(OASA)正式成立,启明星辰、绿盟、360 等 23 家厂商重磅加入
    7月28日,由启明星辰、绿盟、360、阿里云、统信软件、浪潮信息、中兴通讯|中兴新支点、Intel、中科院软件所等23家单位共同发起的龙蜥社区安全联盟(OASA,OpenAnolisSecurityAlliance)(以下简称“安全联盟”),于北京举办了线下启动会,共计33位代表出席。会上,首届安全联盟主席龙勤重点解......
  • VINKA防干扰/抗电压波动的3键/3通道触摸触控芯片VK3603/VK36N3D/VK36N3B 适用于厨房秤
    1.概述VK3603具有3个触摸按键,可用来检测外部触摸按键上人手的触摸动作。该芯片具有较高的集成度,仅需极少的外部组件便可实现触摸按键的检测。提供了3路直接输出功能。芯片内部采用特殊的集成电路,具有高电源电压抑制比,可减少按键检测错误的发生,此特性保证在不利环境条件的应用......
  • AMEYA360:大唐恩智浦荣获〖2023中国具有投资价值车规级芯片企业〗
       2023年8月10日,由无锡市滨湖区政府、市工业和信息化局主办的2023中国汽车半导体新生态论坛暨第五届太湖创芯峰会在无锡隆重举行。峰会介绍当天,滨湖区“集成电路产教融合与人才培养基地”成功揭牌,《2023全球智能汽车产业图谱及研究报告》同期发布。期间,在无锡市......
  • AMEYA360:DNB1101大唐恩智浦工规级电池管理芯片
    大唐恩智浦作为全球领先的半导体供应商,一直致力于为全球客户提供高质量的解决方案。在电池管理芯片领域,大唐恩智浦推出的DNB1101可谓是一款工规级的电池管理芯片,其卓越的性能和可靠性成为市场上备受全球领先的半导体供应商,一直致力于为全球客户提供高质量的解决方案。在电池管......
  • i5 13600K和i5 12600k差距 i513600K和12600k对比
    i5-13600K(viaPassMark),这款RaptorLakeB0芯片采用了6P+8E(14C/20T)的混合式CPU核心设计。组装电脑选i513600K还是i512600K怎么搭配更合适这些点很重要http://www.adiannao.cn/du尽管大核数量保持不变,但架构从GoldenCove升级到了RaptorCove、且小核数量也较上一......