2023-05-11:给你一个 m x n 的二进制矩阵 grid, 每个格子要么为 0 (空)要么为 1 (被占据), 给你邮票的尺寸为 stampHeight x stampWidth。 我们想将

2023-05-11:给你一个 m x n 的二进制矩阵 grid,

每个格子要么为 0 (空)要么为 1 (被占据),

给你邮票的尺寸为 stampHeight x stampWidth。

我们想将邮票贴进二进制矩阵中,且满足以下 限制 和 要求 :

覆盖所有空格子,不覆盖任何被占据的格子,

可以放入任意数目的邮票,邮票可以相互有重叠部分,

邮票不允许旋转,邮票必须完全在矩阵内,

如果在满足上述要求的前提下,可以放入邮票,请返回 true ,否则返回 false。

输入:grid = [[1,0,0,0],[1,0,0,0],[1,0,0,0],[1,0,0,0],[1,0,0,0]], stampHeight = 4, stampWidth = 3。

输出:true。

答案2023-05-11:

大体过程如下:

1.首先对矩阵 grid 进行二维前缀和计算,得到一个新的矩阵 sum。该矩阵中每个位置表示从左上角出发,到该位置形成的子矩阵中所有元素的和。

2.对 grid 中的每个为 0 的位置 (i, j),检查以该位置为左上角的子矩阵是否能够被指定的印章完全覆盖。如果可以,将 diff[i][j] 加 1,diff[i][j+stampWidth] 减 1,diff[i+stampHeight][j] 减 1,diff[i+stampHeight][j+stampWidth] 加 1。这里 diff 矩阵用于记录每个位置的变化量。

3.遍历 grid 中的每一行,使用滚动数组的方式还原 cntpre 数组,并通过它们来计算每列中为 0 的位置的数量。同时,如果某个位置 (i, j) 的值为 0 且它所在列中没有其他的 0,则返回 false;否则返回 true。

时间复杂度为 O(mn),其中 m 和 n 分别表示矩阵 grid 的行数和列数。这是因为函数需要遍历整个矩阵,并对每个位置进行常数次操作。同时,二维前缀和、二维差分和滚动数组优化的时间复杂度也都是 O(mn)。

空间复杂度为 O(mn),因为函数中创建了两个 m+1 行 n+1 列的二维数组 sumdiff,以及一个长度为 n+1 的一维数组 cntpre。这些数组所占用的总空间为 (m+1)(n+1) + 2(n+1) = mn + 3m + 3n + 3,即 O(mn)。

go完整代码如下:

package main

import "fmt"

func main() {
    grid := [][]int{{1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0}}
    stampHeight := 4
    stampWidth := 3
    isPossibleToStamp := possibleToStamp(grid, stampHeight, stampWidth)
    fmt.Println(isPossibleToStamp)
}

func possibleToStamp(grid [][]int, stampHeight, stampWidth int) bool {
    m, n := len(grid), len(grid[0])
    sum := make([][]int, m+1)
    sum[0] = make([]int, n+1)
    diff := make([][]int, m+1)
    diff[0] = make([]int, n+1)
    for i, row := range grid {
        sum[i+1] = make([]int, n+1)
        for j, v := range row { // grid 的二维前缀和
            sum[i+1][j+1] = sum[i+1][j] + sum[i][j+1] - sum[i][j] + v
        }
        diff[i+1] = make([]int, n+1)
    }

    for i, row := range grid {
        for j, v := range row {
            if v == 0 {
                x, y := i+stampHeight, j+stampWidth // 注意这是矩形右下角横纵坐标都 +1 后的位置
                if x <= m && y <= n && sum[x][y]-sum[x][j]-sum[i][y]+sum[i][j] == 0 {
                    diff[i][j]++
                    diff[i][y]--
                    diff[x][j]--
                    diff[x][y]++ // 更新二维差分
                }
            }
        }
    }

    // 还原二维差分矩阵对应的计数矩阵,这里用滚动数组实现
    cnt := make([]int, n+1)
    pre := make([]int, n+1)
    for i, row := range grid {
        for j, v := range row {
            cnt[j+1] = cnt[j] + pre[j+1] - pre[j] + diff[i][j]
            if cnt[j+1] == 0 && v == 0 {
                return false
            }
        }
        cnt, pre = pre, cnt
    }
    return true
}

[图片上传失败...(image-e29b13-1683900654158)]

rust完整代码如下:

fn main() {
    let grid = vec![
        vec![1, 0, 0, 0],
        vec![1, 0, 0, 0],
        vec![1, 0, 0, 0],
        vec![1, 0, 0, 0],
        vec![1, 0, 0, 0],
    ];
    let stamp_height = 4;
    let stamp_width = 3;
    let is_possible_to_stamp = possible_to_stamp(&grid, stamp_height, stamp_width);
    println!("{}", is_possible_to_stamp);
}

fn possible_to_stamp(grid: &[Vec<i32>], stamp_height: usize, stamp_width: usize) -> bool {
    let m = grid.len();
    let n = grid[0].len();
    let mut sum = vec![vec![0; n + 1]; m + 1];
    let mut diff = vec![vec![0; n + 1]; m + 1];

    for i in 0..m {
        for j in 0..n {
            sum[i + 1][j + 1] = sum[i + 1][j] + sum[i][j + 1] - sum[i][j] + grid[i][j];
        }
    }

    for i in 0..m {
        for j in 0..n {
            if grid[i][j] == 0 {
                let x = i + stamp_height;
                let y = j + stamp_width;

                if x <= m && y <= n && sum[x][y] - sum[x][j] - sum[i][y] + sum[i][j] == 0 {
                    diff[i][j] += 1;
                    diff[i][y] -= 1;
                    diff[x][j] -= 1;
                    diff[x][y] += 1;
                }
            }
        }
    }

    let mut cnt = vec![0; n + 1];
    let mut pre = vec![0; n + 1];

    for i in 0..m {
        for j in 0..n {
            cnt[j + 1] = cnt[j] + pre[j + 1] - pre[j] + diff[i][j];

            if cnt[j + 1] == 0 && grid[i][j] == 0 {
                return false;
            }
        }

        std::mem::swap(&mut cnt, &mut pre);
    }

    true
}

[图片上传失败...(image-3dcba8-1683900654158)]

c语言完整代码如下:

#include <stdio.h>
#include <stdlib.h>

int possibleToStamp(int** grid, int gridSize, int* gridColSize, int stampHeight, int stampWidth) {
    int m = gridSize, n = *gridColSize;
    int** sum = (int**)malloc(sizeof(int*) * (m + 1));
    for (int i = 0; i <= m; i++) {
        sum[i] = (int*)malloc(sizeof(int) * (n + 1));
    }
    int** diff = (int**)malloc(sizeof(int*) * (m + 1));
    for (int i = 0; i <= m; i++) {
        diff[i] = (int*)malloc(sizeof(int) * (n + 1));
    }

    for (int i = 0; i < m; i++) {
        for (int j = 0; j < n; j++) {
            sum[i + 1][j + 1] = sum[i + 1][j] + sum[i][j + 1] - sum[i][j] + grid[i][j];
        }
    }

    for (int i = 0; i < m; i++) {
        for (int j = 0; j < n; j++) {
            if (grid[i][j] == 0) {
                int x = i + stampHeight, y = j + stampWidth;
                if (x <= m && y <= n && sum[x][y] - sum[x][j] - sum[i][y] + sum[i][j] == 0) {
                    diff[i][j]++;
                    diff[i][y]--;
                    diff[x][j]--;
                    diff[x][y]++;
                }
            }
        }
    }

    int* cnt = (int*)malloc(sizeof(int) * (n + 1));
    int* pre = (int*)malloc(sizeof(int) * (n + 1));
    for (int i = 0; i <= n; i++) {
        cnt[i] = 0;
        pre[i] = 0;
    }
    for (int i = 0; i < m; i++) {
        for (int j = 0; j < n; j++) {
            cnt[j + 1] = cnt[j] + pre[j + 1] - pre[j] + diff[i][j];
            if (cnt[j + 1] == 0 && grid[i][j] == 0) {
                free(cnt);
                free(pre);
                for (int k = 0; k <= m; k++) {
                    free(sum[k]);
                }
                free(sum);
                for (int k = 0; k <= m; k++) {
                    free(diff[k]);
                }
                free(diff);
                return 0;
            }
        }
        int* tmp = cnt;
        cnt = pre;
        pre = tmp;
    }

    free(cnt);
    free(pre);
    for (int i = 0; i <= m; i++) {
        free(sum[i]);
    }
    free(sum);
    for (int i = 0; i <= m; i++) {
        free(diff[i]);
    }
    free(diff);
    return 1;
}

int main() {
    int gridSize = 5, gridColSize = 4;
    int** grid = (int**)malloc(sizeof(int*) * gridSize);
    for (int i = 0; i < gridSize; i++) {
        grid[i] = (int*)malloc(sizeof(int) * gridColSize);
    }
    int data[5][4] = { {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0} };
    for (int i = 0; i < gridSize; i++) {
        for (int j = 0; j < gridColSize; j++) {
            grid[i][j] = data[i][j];
        }
    }
    int stampHeight = 4, stampWidth = 3;
    int isPossibleToStamp = possibleToStamp(grid, gridSize, &gridColSize, stampHeight, stampWidth);
    printf("%s\n", isPossibleToStamp ? "true" : "false");
    for (int i = 0; i < gridSize; i++) {
        free(grid[i]);
    }
    free(grid);
    return 0;
}

[图片上传失败...(image-d2f156-1683900654158)]

c++完整代码如下:

#include <iostream>
#include <vector>

using namespace std;

bool possibleToStamp(vector<vector<int>>& grid, int stampHeight, int stampWidth) {
    int m = grid.size(), n = grid[0].size();
    vector<vector<int>> sum(m + 1, vector<int>(n + 1)), diff(m + 1, vector<int>(n + 1));

    for (int i = 0; i < m; i++) {
        for (int j = 0; j < n; j++) {
            sum[i + 1][j + 1] = sum[i + 1][j] + sum[i][j + 1] - sum[i][j] + grid[i][j];
        }
    }

    for (int i = 0; i < m; i++) {
        for (int j = 0; j < n; j++) {
            if (grid[i][j] == 0) {
                int x = i + stampHeight, y = j + stampWidth;
                if (x <= m && y <= n && sum[x][y] - sum[x][j] - sum[i][y] + sum[i][j] == 0) {
                    diff[i][j]++;
                    diff[i][y]--;
                    diff[x][j]--;
                    diff[x][y]++;
                }
            }
        }
    }

    vector<int> cnt(n + 1), pre(n + 1);
    for (int i = 0; i < m; i++) {
        for (int j = 0; j < n; j++) {
            cnt[j + 1] = cnt[j] + pre[j + 1] - pre[j] + diff[i][j];
            if (cnt[j + 1] == 0 && grid[i][j] == 0) {
                return false;
            }
        }
        swap(cnt, pre);
    }

    return true;
}

int main() {
    vector<vector<int>> grid{ {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0}, {1, 0, 0, 0} };
    int stampHeight = 4, stampWidth = 3;
    bool isPossibleToStamp = possibleToStamp(grid, stampHeight, stampWidth);
    cout << (isPossibleToStamp ? "true" : "false") << endl;
    return 0;
}

[图片上传失败...(image-9919f4-1683900654158)]

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,277评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,689评论 3 393
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,624评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,356评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,402评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,292评论 1 301
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,135评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,992评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,429评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,636评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,785评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,492评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,092评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,723评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,858评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,891评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,713评论 2 354

推荐阅读更多精彩内容