Bootstrap

(leetcode算法题)2271. 毯子覆盖的最多白色砖块数

 滑动窗口的窗口长度固定,那么其研究的窗口所在位置一定是通过画图遍历,这样能够快速的分析问题,直接围绕这些特殊的位置求解固定长度的滑动窗口问题

h

下面说明只研究一种状态的窗口,

将窗口的右端点,放在tiles中每个瓷砖的右端点来确定每一个状态,

从图中可以看出,之所以这么选,是因为中间的把蓝条(毯子)的右端点放到瓷砖的右端点,其覆盖的瓷砖的长度总是大于等于把蓝条(毯子的右端点)放到瓷砖的中间,

注意:这里的l 和 r 应该作为tiles的下标,那么l和r在不同位置时代表什么含义?

case1: l == r - k (k > 0)

            此时代表将 l 指向的 tile, l + 1, ... l + k 指向的tile都作为窗口中的内容,不会考虑绿色区域

case2: l == r

            此时代表 只考虑将 r 指向的 tile作为窗口中的内容

这个题目需要问以下几个问题

Q1. 什么情况需要考虑开始从 r 所指的位置循环执行进窗口动作?

        r 遍历到的当前tiles, 需要不断进窗口

Q2. 什么时候需要暂停执行进窗口动作?

       r 遍历到的当前tiles, 需要不断进窗口

Q3. 什么时候需要考虑开始从 l 所指的位置循环执行出窗口动作?

       l 指向的tiles 的右端点 大于 r指向的tiles的右端点 - 毯子的长度 + 1

Q4. 什么时候需要暂停执行出窗口动作?

       l 指向的tiles 的右端点 小于等于 r指向的tiles的右端点 - 毯子的长度 + 1

Q5. 什么时候更新结果?

       每次进窗口动作执行一次之后,执行以下逻辑来更新结果

           首先,判断出窗口,

           如果对 l 指向的 tiles 进行出窗口,那么 l 每出一个窗口,

           就用结果将当前的 l 指向的tiles的长度给减去

           其次,在出窗口暂停后,

           如果 毯子从 r 所指向的 tiles的右端点无法覆盖完 l 所指向的 tiles 的左端点

           那么用结果将 减去 α,其中α分两种情况

           case1: r 指向的tiles的右端点 - 毯子的长度 > l 所指向的 tiles 的左端点,也就是说毯子右端

                      点在r 指向的右端点,左端点在l 所指向的 tiles 的中间

                      此时 α = r 指向的tiles的右端点 - 毯子的长度 - l 所指向的 tiles 的左端点 + 1

           case2: r 指向的tiles的右端点 - 毯子的长度 < l 所指向的 tiles 的左端点,也就是说毯子右端

                      点在r 指向的右端点,左端点在l 所指向的 tiles 的左端点的左边,但是不在 l - 1 所指

                      向的tiles 的右端点的左边,此时 α = 0

                以tiles = [[1,5],[10,11],[12,18],[20,25],[30,32]], carpetLen = 9为例(注意和原题示例不同)

下图中黑色叉号表明的是出窗口要删除的瓷砖长度,

棕色叉号表明的是当蓝色的毯子盖不住从 l 指向tiles的左端点到l指向的tiles的右端点时,需要删除的长度           

                

        ​​​​​​​        

// 3413. 收集连续 K 个袋子可以获得的最多硬币数量 和 2271. 毯子覆盖的最多白色砖块数
// 用相同的滑动窗口算法
class Solution {
public:
    int maximumWhiteTiles(vector<vector<int>>& tiles, int carpetLen) {
        int n = tiles.size();
        sort(tiles.begin(), tiles.end(), [&](vector<int>& v1, vector<int>& v2){
            return v1[0] < v2[0];
        });
        int ret = 0, left = 0, curlen = 0;
        for(int i = 0; i < n; ++i){
            int tl = tiles[i][0], tr = tiles[i][1];
            curlen += (tr - tl + 1);
            while(tiles[left][1] < tr - carpetLen + 1){
                curlen -= (tiles[left][1] - tiles[left][0] + 1);
                left++;
            }
            int uncover = max(0, tr - carpetLen - tiles[left][0] + 1);
            ret = max(ret, curlen - uncover);
        }

        return ret;
    }
};

 

;