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 中的每一行,使用滚动数组的方式还原 cnt 和 pre 数组,并通过它们来计算每列中为 0 的位置的数量。同时,如果某个位置 (i, j) 的值为 0 且它所在列中没有其他的 0,则返回 false;否则返回 true。
时间复杂度为 O(mn),其中 m 和 n 分别表示矩阵 grid 的行数和列数。这是因为函数需要遍历整个矩阵,并对每个位置进行常数次操作。同时,二维前缀和、二维差分和滚动数组优化的时间复杂度也都是 O(mn)。
空间复杂度为 O(mn),因为函数中创建了两个 m+1 行 n+1 列的二维数组 sum 和 diff,以及一个长度为 n+1 的一维数组 cnt 和 pre。这些数组所占用的总空间为 (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
}

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
}

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;
}

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;
}

2023-05-11:给你一个 m x n 的二进制矩阵 grid, 每个格子要么为 0 (空)要么为 1 (被占据), 给你邮票的尺寸为 stampHeight x stampWidth。 我们想将的更多相关文章
- 螺旋填数:读入两个整数m,n,输出一个m行n列的矩阵,这个矩阵是1~m*n这些自然数按照右、下、左、上螺旋填入的结果。
package Day8_06; /*读入两个整数m,n,输出一个m行n列的矩阵,这个矩阵是1~m*n这些自然数按照右.下.左.上螺旋填入的结果. * 例如读入数字4,5,则输出结果为: * 1 2 ...
- 一个unsigned int 数的二进制表示中有多少个1
这是一道面试题可以用以下的一些方案.第一种是很容易想到的采用循环的方式并且与1进行位与运算,具体代码如下. 1unsigned int GetBitNumOfOne_ByLoop1(unsigned ...
- 《Python CookBook2》 第一章 文本 - 过滤字符串中不属于指定集合的字符 && 检查一个字符串是文本还是二进制
过滤字符串中不属于指定集合的字符 任务: 给定一个需要保留的字符串的集合,构建一个过滤函数,并可将其应用于任何字符串s,函数返回一个s的拷贝,该拷贝只包含指定字符集合中的元素. 解决方案: impor ...
- java语言将任意一个十进制数数字转换为二进制形式,并输出转换后的结果
package com.llh.demo; import java.util.Scanner; /** * * @author llh * */ public class Test { /* * 将任 ...
- 用最小的空间复杂度找出一个长度为n的数组且数据中的元素是[0,n-1]中任一个重复的数据。
用最小的空间复杂度找出一个长度为n的数组且数据中的元素是[0,n-1]中任一个重复的数据. 比如:[1, 2, 3, 3, 2, 2, 6, 7, 8, 9] 中 2 or 3 分析:这道题目,实现比 ...
- Java学习笔记 11/15:一个简单的JAVA例子
首先来看一个简单的 Java 程序. 来看下面这个程序,试试看是否看得出它是在做哪些事情! 范例:TestJava.java // TestJava.java,java 的简单范例 public ...
- 2018/05/11 PHP 设计模式之 适配器模式
什么是适配器模式? 简单来说,我想买一根充电线,我买一根安卓的?还是买一根苹果的? 我也不确定,因为我以可能会换手机,对于我的形式我也不确定. 所以,我要买一根可以同时适配 安卓/苹果 的线. 所谓适 ...
- 2016/05/11 Thinkphp 3.2.2 验证码 使用 及校验
先新建一个公共控制器,用于放置验证码的实例化代码(不用新建控制器也行,任意公共控制器都可以). 例如:PublicController.class.php 4 5 6 7 8 9 10 11 12 1 ...
- 剑指offer19:按照从外向里以顺时针的顺序依次打印出每一个数字,4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印出数字1,2,3,4,8,12,16,15,14,13,9,5,6,7,11,10.
1 题目描述 输入一个矩阵,按照从外向里以顺时针的顺序依次打印出每一个数字,例如,如果输入如下4 X 4矩阵: 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 则依次打印 ...
- 05 . Go+Vue开发一个线上外卖应用(Session集成及修改用户头像到Fastdfs)
用户头像上传 功能介绍 在用户中心中,允许用户更换自己的头像.因此,我们开发上传一张图片到服务器,并保存成为用户的头像. 接口解析 在用户模块的控制器MemberController中,解析头像上传的 ...
随机推荐
- 文本的格式化标签(粗体,斜体)和 <div>和<span>标签(都是双标签)
上一个笔记有提到各种型号的标题,为了保证文章的美观,又会有除了标题之外的东西,比如粗体,斜体,下划线,删除线和各种分隔 1加粗,<strong><strong/>或者<b ...
- Javaweb知识复习--MyBatis+Mapper代理开发
一种持久层框架,主要用于简化JDBC MyBatis应用步骤 1.在数据库里面创建一个表 2.创建模块,导入坐标 就是新建一个Maven项目,在pom.xml里面导入mybatis相应导包依赖代码: ...
- nat是干什么的,为什么要有nat?以及谈谈ovs里使用ct实现nat功能
博客竟然不显示更新的时间,只有个发布时间.看起来像2个月没更新一样,其实更新了几行呢.好几个东西想理一下,本来想和周记放一起了,但放一起就没有主题了. 当然一搜也有一些很好的博客,更详细:https: ...
- 基于对象的实时空间音频渲染丨Dev for Dev 专栏
本文为「Dev for Dev 专栏」系列内容,作者为声网音频算法工程师 李嵩. 随着元宇宙概念的引入,空间音频这项技术慢慢映入大家的眼帘.关于空间音频的基础原理,我们做过一期科普视频 -- 「空间音 ...
- Java面试——Tomcat
更多内容,前往个人博客 一.Tomcat 顶层架构 Tomcat 中最顶层的容器是 Server,代表着整个服务器,从上图中可以看出,一个 Server可以包含至少一个 Service,用于具体提 ...
- 玩转Mybatis高级特性:让你的数据操作更上一层楼
目录 动态SQL 缓存机制 插件机制 自定义类型转换 总结 Mybatis高级特性能够帮助我们更加灵活地操作数据库,包括动态SQL.缓存机制.插件机制.自定义类型转换等.学习这些特性可以让我们更好地利 ...
- R数据分析:生存分析的列线图的理解与绘制详细教程
列线图作为一个非常简单明了的临床辅助决策工具,在临床中用的(发文章的)还是比较多的,尤其是肿瘤预后: Nomograms are widely used for cancer prognosis, p ...
- 3d基础 - 从模型坐标到屏幕坐标
在 3D 引擎中,场景通常被描述为三维空间中的模型或对象,每个模型对象由许多三维顶点组成.最终,这些模型对象将在平面屏幕上呈现和显示. 渲染场景始终相对于摄像机,因此,还必须相对于摄像机的视图定义场景 ...
- 【过滤器设计模式详解】C/Java/JS/Go/Python/TS不同语言实现
简介 过滤器模式(Filter Pattern)或标准模式(Criteria Pattern),是一种结构型模式.这种模式允许使用不同的标准条件来过滤一组对象,并通过逻辑运算的方式把各条件连接起来,它 ...
- 在 Kubernetes 集群上部署 VSCode
在 Kubernetes 集群上部署 VSCode Visual Studio Code Visual Studio Code 是一个轻量级但功能强大的源代码编辑器,可在您的桌面上运行,适用于 Win ...