避免洪水泛滥(leetcode1448)

问题描述:

你的国家有无数个湖泊,所有湖泊一开始都是空的。当第 n 个湖泊下雨的时候,如果第 n 个湖泊是空的,那么它就会装满水,否则这个湖泊会发生洪水。你的目标是避免任意一个湖泊发生洪水。

给你一个整数数组 rains ,其中:

rains[i] > 0 表示第 i 天时,第 rains[i] 个湖泊会下雨。
rains[i] == 0 表示第 i 天没有湖泊会下雨,你可以选择 一个 湖泊并 抽干 这个湖泊的水。
请返回一个数组 ans ,满足:

ans.length == rains.length
如果 rains[i] > 0 ,那么ans[i] == -1 。
如果 rains[i] == 0 ,ans[i] 是你第 i 天选择抽干的湖泊。
如果有多种可行解,请返回它们中的 任意一个 。如果没办法阻止洪水,请返回一个 空的数组 。

请注意,如果你选择抽干一个装满水的湖泊,它会变成一个空的湖泊。但如果你选择抽干一个空的湖泊,那么将无事发生(详情请看示例 4)。

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

示例 1:

输入:rains = [1,2,3,4]
输出:[-1,-1,-1,-1]
解释:第一天后,装满水的湖泊包括 [1]
第二天后,装满水的湖泊包括 [1,2]
第三天后,装满水的湖泊包括 [1,2,3]
第四天后,装满水的湖泊包括 [1,2,3,4]
没有哪一天你可以抽干任何湖泊的水,也没有湖泊会发生洪水。
示例 2:

输入:rains = [1,2,0,0,2,1]
输出:[-1,-1,2,1,-1,-1]
解释:第一天后,装满水的湖泊包括 [1]
第二天后,装满水的湖泊包括 [1,2]
第三天后,我们抽干湖泊 2 。所以剩下装满水的湖泊包括 [1]
第四天后,我们抽干湖泊 1 。所以暂时没有装满水的湖泊了。
第五天后,装满水的湖泊包括 [2]。
第六天后,装满水的湖泊包括 [1,2]。
可以看出,这个方案下不会有洪水发生。同时, [-1,-1,1,2,-1,-1] 也是另一个可行的没有洪水的方案。
示例 3:

输入:rains = [1,2,0,1,2]
输出:[]
解释:第二天后,装满水的湖泊包括 [1,2]。我们可以在第三天抽干一个湖泊的水。
但第三天后,湖泊 1 和 2 都会再次下雨,所以不管我们第三天抽干哪个湖泊的水,另一个湖泊都会发生洪水。
示例 4:

输入:rains = [69,0,0,0,69]
输出:[-1,69,1,1,-1]
解释:任何形如 [-1,69,x,y,-1], [-1,x,69,y,-1] 或者 [-1,x,y,69,-1] 都是可行的解,其中 1 <= x,y <= 10^9
示例 5:

输入:rains = [10,20,20]
输出:[]
解释:由于湖泊 20 会连续下 2 天的雨,所以没有没有办法阻止洪水。

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

大体思路:

一个贪心思路,遇到晴天时,应该优先选择里当天之后最近一次再次装满水的湖泊抽干,以示例2为例,

rains = [1,2,0,0,2,1]

第三天是个晴天,发现里它最近再次装满水的湖泊是2号湖泊,所以应该优先抽干二号湖泊的水。

解法一:直接贪心

使用一Set存储已经装满水的湖泊号,遍历过程中若当天下雨,就判断set中是否存在该湖泊,若存在则说明该湖泊没能抽干,引发浑水;否则将下雨的湖泊号存入set。若当前晴天,就在从当天的下一天开始遍历,找到第一个再次下雨的湖泊,并将其从set中删除。该方法O(N^2),直接超时了...代码如下:

class Solution {
    public int[] avoidFlood(int[] rains) {
        int[] result = new int[rains.length];
        Set<Integer> set = new HashSet<>();
        for(int i = 0; i < rains.length; i++){
            if(rains[i] != 0){
                if(set.contains(rains[i])){
                    return new int[0];
                }
                result[i] = -1;
                set.add(rains[i]);
                continue;
            }
            result[i] = 1;//随便抽一个
            for(int j = i + 1; j < rains.length; j++){
                if(set.contains(rains[j])){
                    result[i] = rains[j];
                    set.remove(rains[j]);
                    break;
                }
            }
        }
        return result;
    }
}

解法二:贪心 + 二分

首先使用一个list存储所有晴天的日期(从前至后)。使用一Map存储下雨的湖泊号和日期(key = 下雨湖泊号 value = 日期)。

遍历过程中,若当天下雨,判断该湖泊之前是否下雨,若下之前亦下过雨,则此时在晴天list中二分查找一个距上次下雨之后最近的晴天(大于target值中最小的元素),并在该天对该湖泊放水,若找不到则证明发洪水了,最后更新map;若当天不下雨则将当天日期存入list。时间复杂度O(Nlog(N)),实现代码如下:

class Solution {
    public int[] avoidFlood(int[] rains) {
        int[] result = new int[rains.length];
        List<Integer> sunday = new ArrayList<>();
        Map<Integer, Integer> map = new HashMap<>(); // key = 下雨湖泊号 value = 日期
        for(int i = 0; i < rains.length; i++){
            if(rains[i] == 0){
                sunday.add(i);
                result[i] = 1;
                continue;
            }
            result[i] = -1;
            if(map.containsKey(rains[i])){
                int pre = map.get(rains[i]);
                if(sunday.isEmpty() || sunday.get(sunday.size() - 1) < pre){
                    return new int[0];
                }
                int index = find(sunday, pre);
                result[sunday.remove(index)] = rains[i];
            }
            map.put(rains[i], i);
        }
        return result;
    }
    public int find(List<Integer> list, int target){ //找到大于target的第一元素
        int left = 0, right = list.size() - 1;
        while(left < right - 1){
            int mid = left + (right - left) / 2;
            if(list.get(mid) > target){
                right = mid;
            }else{
                left = mid;
            }
        }
        if(list.get(left) > target){
            return left;
        }
        return right;
    }
}

解法三:贪心 + 最小堆

解法一中我们知道,遇到晴天时我们应该找到其之后的再次满水的湖泊,对其放水。因此使用一名为nextRain的数组存储下次下雨的时间,nextRain[i] 为rain[i]湖泊下次下雨的日期;使用一map存储湖泊状态(是否有水);使用一最小堆存储下一次下雨时间。

遍历过程中若当天有雨,则判断当天下雨的湖泊之前是否有水,有水则说明已经发洪水了,否则将当天nextRain[i]存入最小堆,;遇到晴天时,从最小堆中弹出一个元素,该元素时里当天最近的再次下雨的日期,因此对该日期对应的湖泊放水即可。

实现代码如下:(时间复杂度:O(Nlog(N)))

class Solution {
    public int[] avoidFlood(int[] rains) {
        int[] nextRain = new int[rains.length]; // 存储rain[i] 位置下次下雨的时间
        Map<Integer, Integer> map = new HashMap<>();
        for(int i = rains.length - 1; i >= 0; i--){
            if(rains[i] != 0){
                nextRain[i] = map.getOrDefault(rains[i], Integer.MAX_VALUE);
                map.put(rains[i], i);
            }
        }
        Map<Integer, Boolean> hasWater = new HashMap<>();
        Queue<Integer> minHeap = new PriorityQueue<>();
        int[] result = new int[rains.length];
        for(int i = 0; i < rains.length; i++){
            if(rains[i] != 0){
                result[i] = -1;
                if(hasWater.getOrDefault(rains[i], false)){
                    return new int[0];
                }
                hasWater.put(rains[i], true);
                minHeap.add(nextRain[i]);
            }else{
                if(minHeap.isEmpty() || minHeap.peek() == Integer.MAX_VALUE){// 随便抽一个
                    result[i] = 1;
                }else{
                    int num = rains[minHeap.remove()];
                    result[i] = num;
                    hasWater.put(num, false);
                }
            }
        }
         return result;
    }
}

 

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值