LeetCode-1326-灌溉花园的最少水龙头数目

文章介绍了两种解决区间覆盖问题的方法:一是使用贪心算法,通过维护一个记录最远右端点的数组,动态更新并判断能否覆盖所有区间;二是采用动态规划,构建状态转移方程,寻找覆盖[0,i]区间所需的最少节点数。两种方法均关注于减少水龙头的最小数量来覆盖整个区域。
摘要由CSDN通过智能技术生成

在这里插入图片描述

1、贪心

我们利用数组rightMost[n]记录以节点n为左端点的子区间中最远右端点的位置。而后我们对每个节点i进行判断,记左端点不超过i的所有子区间中右端点的最远位置为last,利用pre记录在上一个子区间的结束位置。每当我们抵达新位置i时,都对last进行更新:1、若last=i,说明下个位置无法覆盖,返回-1;2、若pre=i,说明我们使用完当前子区间,我们更新pre并将子区间数加一。

class Solution {
public:
    int minTaps(int n, vector<int>& ranges) {
        vector<int> rightMost(n + 1);
        iota(rightMost.begin(), rightMost.end(), 0);
        for (int i = 0; i <= n; i++) {
            int start = max(0, i - ranges[i]);
            int end = min(n, i + ranges[i]);
            rightMost[start] = max(rightMost[start], end);
        }
        int last = 0, ret = 0, pre = 0;
        for (int i = 0; i < n; i++) {
            last = max(last, rightMost[i]);
            if (i == last) {
                return -1;
            }
            if (i == pre) {
                ret++;
                pre = last;
            }
        }
        return ret;
    }
};

2、动态规划法

利用dp[i]记录覆盖区间[0,i]所需要的最少节点数,状态转移公式为: d p [ i ] = m i n ( d p [ i ] , 1 + d p [ s t a r t j ] ) dp[i]=min(dp[i],1+dp[start_j]) dp[i]=min(dp[i],1+dp[startj])

class Solution {
public:
    int minTaps(int n, vector<int>& ranges) {
        vector<pair<int, int>> intervals;
        for (int i = 0; i <= n; i++) {
            int start = max(0, i - ranges[i]);
            int end = min(n, i + ranges[i]);
            intervals.emplace_back(start, end);
        }
        sort(intervals.begin(), intervals.end());
        vector<int> dp(n + 1, INT_MAX);
        dp[0] = 0;
        for (auto [start, end] : intervals) {
            if (dp[start] == INT_MAX) {
                return -1;
            }
            for (int j = start; j <= end; j++) {
                dp[j] = min(dp[j], dp[start] + 1);
            }
        }
        return dp[n];
    }
};
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值