leetcode 1962. 移除石子使总数最小

题目描述:

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

选出任一石子堆 piles[i] ,并从中 移除 floor(piles[i] / 2) 颗石子。
注意:你可以对 同一堆 石子多次执行此操作。

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

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

 

示例 1:

输入:piles = [5,4,9], k = 2
输出:12
解释:可能的执行情景如下:
- 对第 2 堆石子执行移除操作,石子分布情况变成 [5,4,5] 。
- 对第 0 堆石子执行移除操作,石子分布情况变成 [3,4,5] 。
剩下石子的总数为 12 。
示例 2:

输入:piles = [4,3,6,7], k = 3
输出:12
解释:可能的执行情景如下:
- 对第 2 堆石子执行移除操作,石子分布情况变成 [4,3,3,7] 。
- 对第 3 堆石子执行移除操作,石子分布情况变成 [4,3,3,4] 。
- 对第 0 堆石子执行移除操作,石子分布情况变成 [2,3,3,4] 。
剩下石子的总数为 12 。
 

提示:

1 <= piles.length <= 105
1 <= piles[i] <= 104
1 <= k <= 105

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/remove-stones-to-minimize-the-total
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

解题思路:

1、要使最后剩余的石子数最少,那每次减少的应该是当前石头数目最多的一堆,问题转换为求数组中的最大值问题

2、首先明确最后要求的只是剩余的石头总数,所以顺序可以改变,那么就可以对数组中元素进行排序再找最大值。

3、排序的话就涉及到了排序算法:

  由于数组定义后不能修改大小,首先想到把数组转换为list或queue

  (1)List的父类是Collection,继承父类的排序方法,而Collection.sort()采用的排序算法是一种改进的归并排序,归并算法的时间复杂度是O(nlog n)

  (2)使用Queue构建大顶堆,使用优先级队列,比较实现的是Comparator接口。堆中添加元素时间复杂度O(log N)

4、采用优先级队列构造大顶堆,每次取堆顶元素,减少一半后再加入堆中,循环k次得到最后的堆

5、将队列中所有元素相加得到最后结果。

public class RemoveStone {
    //元素可以改变顺序!!超时!!
    public int minStoneSum(int[] piles, int k) {
        for(int i = 0; i < k; i++){
            int max_number = 0;//当前轮最大的数值
            int max_number_index = 0;//当前值最大的下标
            for(int j = 0; j < piles.length; j++ ){
                if(piles[j] > max_number){
                    max_number_index = j;
                    max_number = piles[j];
                }
            }
            //去除当前值最的number的一半
            piles[max_number_index] -= (int) Math.floor(piles[max_number_index]/2);
        }
        int sum = 0;
        for(int i = 0; i < piles.length; i++){
            sum += piles[i];
        }
        return sum;
    }

    public int minStoneSum1(int[] piles, int k) {
        PriorityQueue<Integer> pq = new PriorityQueue<>((a,b) -> b-a);//构建大顶堆
        //将piles中元素入队
        for(int i = 0; i < piles.length; i++){
            pq.add(piles[i]);
        }
        //取堆顶元素减少一半并且重新放会堆中,循环k次
        //堆添加元素时间复杂度O(log N)
        for(int j = 0; j < k; j++){
            int top_que = pq.poll();
            top_que -= Math.floor(top_que/2.0);
            pq.add(top_que);
        }
        //计算和
        int sum = 0;
        for(Integer q: pq){
            sum += q;
        }
        return sum;
    }
    public int minStoneSum2(int[] piles, int k) {
        List<Integer> piles_list = new ArrayList<>();
        //将数组转换为list  超时!!
        for(int i = 0; i< piles.length;i++){
            piles_list.add(piles[i]);
        }

        //Collection.sort()采用的排序算法是一种改进的归并排序,归并算法的时间复杂度是O(nlog n)
        for(int j = 0; j < k; j++){
            piles_list.sort((a,b)->b-a);//先对list进行倒序排序
            piles_list.set(0, piles_list.get(0)-(int) Math.floor(piles_list.get(0)/2.0));//设置第一个值减少原来的一半
        }
        int sum =0;
        for(Integer q : piles_list){
            sum += q;
        }
        return sum;
    }
    public static void main(String[] args) {
        int[] piles = new int[]{4,3,6,7};
        int k = 3;
        RemoveStone rs = new RemoveStone();
        int result = rs.minStoneSum2(piles, k);
        System.out.println(result);
    }
}

 

posted @ 2021-09-16 20:56  做个读书人  阅读(69)  评论(0编辑  收藏  举报