Leecode 周赛318场

Leecode 周赛318场

第一题

直接遍历

code

    public int[] applyOperations(int[] nums) {
        for (int i = 0; i < nums.length -1; i++) {
            if (nums[i] == nums[i+1]){
                nums[i] = nums[i+1]*2;
                nums[i+1]=0;
                i++;
            }else{
                continue;
            }
        }

        int[] res = new int[nums.length];
        int index = 0;
        for (int i = 0; i < nums.length; i++) {
            if (nums[i] != 0){
                res[index++] = nums[i];
            }
        }

        return res;
    }

第二题 长度为 K 子数组中的最大和

​ 思路使用滑动窗口,首先创建一些基础变量,使用set来保证子数组没有重复

        long res = 0;       //最终返回值
        int l= 0;           //左指针
        int r=0;            //右指针
        long sum =0;        //总和
        int n = nums.length;; //长度
        Set<Integer> set = new HashSet<>(); //set用来保用子数组无重复

如果我们的元素在set中不存在,那么我们把当前元素放入set集合中,放入成功查看子数组的长度是否大于k,如果大于k那么就移除子数组最左侧的元素,之后判断我们的子数组长度是否等于k,如果满足就判断当前的sum是否为最大值

            if (!set.contains(nums[r])){
                set.add(nums[r]);
                sum += nums[r];

                if (r-l+1 > k){
                    sum-=nums[l];
                    set.remove(nums[l++]);
                }

                if (r-l+1 == k){
                    res = Math.max(res,sum);
                }
                r++;
            }

如果在set集合中已经存在此元素,如果l在r左侧,我们要保证l必须小于r,并且当前这个元素nums[r]依然存在集合中,那么就移除数组最左侧的元素。

else{
    while (l < r && set.contains(nums[r])){
        sum -= nums[l];
        set.remove(nums[l++]);
    }
}

code

public long maximumSubarraySum(int[] nums, int k) {
    long res = 0;       //最终返回值
    int l= 0;           //左指针
    int r=0;            //右指针
    long sum =0;        //总和
    int n = nums.length;; //长度
    Set<Integer> set = new HashSet<>(); //set用来保用子数组无重复

    while (r < n){
        //查看是否已经添加
        if (!set.contains(nums[r])){
            set.add(nums[r]);
            sum += nums[r];

            if (r-l+1 > k){
                sum-=nums[l];
                set.remove(nums[l++]);
            }

            if (r-l+1 == k){
                res = Math.max(res,sum);
            }
            r++;
        }else{
            while (l < r && set.contains(nums[r])){
                sum -= nums[l];
                set.remove(nums[l++]);
            }
        }
    }

    return res;
}

第三题 雇佣 K 位工人的总代价

​ 使用两个优先队列分别维护前candidates个,和后candidates个。

        int l = 0; //数组左侧的指针
		int r= costs.length -1; //数组右侧的指针
        long res = 0;	//res
        int n = costs.length;	
        PriorityQueue<Integer> leftQue = new PriorityQueue<>();
        PriorityQueue<Integer> rigthQue = new PriorityQueue<>();

接下来我们对我们的优先队列进行从初始化,把前candidates个和后candidates个都放入优先队列中

        for (int i = 0; i < candidates; i++) {
             if (l<=r){
                 leftQue.offer(costs[l++]);
             }
             if (l <= r){
                 rigthQue.offer(costs[r--]);
             }
        }

接下来判断下leftQue和rigthQue最小的元素谁是最小的,如果谁的优先队列的元素是最小的谁poll一个,poll完事以后查看是否可以装入新元素,如果l<r就可以装入新的元素,最后记录下总和即可

        for (int i = 0; i < k; i++) {
            int lmin = leftQue.isEmpty() ? Integer.MAX_VALUE: leftQue.peek();
            int rmin = rigthQue.isEmpty() ? Integer.MAX_VALUE : rigthQue.peek();
            if (lmin <= rmin){
                res +=leftQue.poll();
                if (l <= r){
                    leftQue.offer(costs[l++]);
                }

            }else{
                res += rigthQue.poll();
                if (l <= r){
                    rigthQue.offer(costs[r--]);
                }
            }
        }

code

    public long totalCost(int[] costs, int k, int candidates) {
        int l = 0 , r= costs.length -1;
        long res = 0;
        int n = costs.length;
        PriorityQueue<Integer> leftQue = new PriorityQueue<>();
        PriorityQueue<Integer> rigthQue = new PriorityQueue<>();

        for (int i = 0; i < candidates; i++) {
             if (l<=r){
                 leftQue.offer(costs[l++]);
             }
             if (l <= r){
                 rigthQue.offer(costs[r--]);
             }
        }

        for (int i = 0; i < k; i++) {
            int lmin = leftQue.isEmpty() ? Integer.MAX_VALUE: leftQue.peek();
            int rmin = rigthQue.isEmpty() ? Integer.MAX_VALUE : rigthQue.peek();
            if (lmin <= rmin){
                res +=leftQue.poll();
                if (l <= r){
                    leftQue.offer(costs[l++]);
                }

            }else{
                res += rigthQue.poll();
                if (l <= r){
                    rigthQue.offer(costs[r--]);
                }
            }
        }


        return res;
    }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

哇塞大嘴好帅(DaZuiZui)

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

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

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

打赏作者

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

抵扣说明:

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

余额充值