原地建堆,LeetCode 1962. 移除石子使总数最小

一、题目

1、题目描述

给你一个整数数组 piles ,数组 下标从 0 开始 ,其中 piles[i] 表示第 i 堆石子中的石子数量。另给你一个整数 k ,请你执行下述操作 恰好 k 次:

  • 选出任一石子堆 piles[i] ,并从中 移除 floor(piles[i] / 2) 颗石子。

注意:你可以对 同一堆 石子多次执行此操作。

返回执行 k 次操作后,剩下石子的 最小 总数。

floor(x) 为 小于 或 等于 x 的 最大 整数。(即,对 x 向下取整)。

2、接口描述

class Solution {
public:
    int minStoneSum(vector<int>& piles, int k) {

    }
};

3、原题链接

1962. 移除石子使总数最小


二、解题报告

1、思路分析

显然我们每次都移除数目最多的那一堆,最后一定能够满足剩下的最小,每次找到最大数目的石子堆我们可以选择使用大根堆来实现。

我们如果将数组元素都插入堆中,那么需要O(nlogn)的时间复杂度和O(n)的空间复杂度,我们可以选择直接原地建堆,即在原数组上建堆,采用向下调整算法自下而上建堆可以达到O(n)的时间复杂度建堆,关于堆的两种构建算法以及时间复杂度证明,见:堆/二叉堆详解[C/C++]-CSDN博客

然后执行k次操作,如果堆顶元素为1,那么此时无法再拿出元素,我们直接退出循环即可

2、复杂度

时间复杂度:O(n + klogn) 空间复杂度:O(1)

3、代码详解

​C++版本
class Solution {
public:
    int minStoneSum(vector<int>& piles, int k) {
        //原地堆化默认大根堆
        make_heap(piles.begin() , piles.end());
        while(k-- && (piles[0] ^ 1)){
            pop_heap(piles.begin() , piles.end());//弹出堆顶到末尾
            piles.back() -= piles.back() / 2;
            push_heap(piles.begin() , piles.end());//向上调整插入末尾元素到堆
        }
        return accumulate(piles.begin() , piles.end() , 0);
    }
};
python3版本
class Solution:
    def minStoneSum(self, piles: List[int], k: int) -> int:
        for i in range(len(piles)):
            piles[i] *= -1
        heapify(piles)
        while k and (piles[0] ^ 1):
            heapreplace(piles , piles[0] // 2) # 负数向下取整的绝对值和正数向上取整绝对值一样
            k -= 1
        return -sum(piles)

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

EQUINOX1

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

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

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

打赏作者

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

抵扣说明:

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

余额充值