当前位置: 首页 > news >正文

蓝桥杯-子矩阵

"""
题目来源
https://www.lanqiao.cn/problems/3521/learning/?page=1&first_category_id=1&name=%E5%AD%90%E7%9F%A9%E9%98%B5
"""
import os
import sys
from collections import deque# 请在此输入您的代码
n, m, a, b = map(int, input().split())
datas = []
for i in range(n):datas.append(list(map(int, input().split())))# 记录所有子矩阵区间最大值
def max_value_queue(x):# 记录每一行所有区间长度为b的区间最大值ws = []for i in range(n):nums = datas[i]# w记录当前行所有区间长度为b的区间最大值w = []# 单调队列q = deque()for i, v in enumerate(nums):# 1.入队列while q and v >= nums[q[-1]]:q.pop()q.append(i)# 2.出队列# 单调队列的长度限制为b, 当队列元素个数大于b就需要将队首元素出队if i - q[0] + 1 >= b + 1:q.popleft()# 3.开始记录答案if i >= b - 1:# 单调队列降序, 队首元素为区间最大值w.append(nums[q[0]])ws.append(w)# 记录所有二维区间长度为a宽度为b的区间最大值ans = []# 将ws中的每一列作为一行tmps = [[ws[j][i] for j in range(n)] for i in range(len(ws[0]))]for i in range(len(ws[0])):nums = tmps[i]# w记录当前行所有区间长度为a的区间最大值w = []# 单调队列q = deque()for i, v in enumerate(nums):# 1.入队列while q and v >= nums[q[-1]]:q.pop()q.append(i)# 2.出队列# 单调队列的长度限制为a, 当队列元素个数大于a就需要将队首元素出队if i - q[0] + 1 >= a + 1:q.popleft()# 3.开始记录答案if i >= a - 1:# 单调队列降序, 队首元素为区间最大值w.append(nums[q[0]])ans.append(w)return ans# 记录所有矩阵最小值
def min_value_queue(x):# 记录每一行所有区间长度为b的区间最小值ws = []for i in range(n):nums = datas[i]# w记录当前行所有区间长度为b的区间最小值w = []# 单调队列q = deque()for i, v in enumerate(nums):# 1.入队列while q and v <= nums[q[-1]]:q.pop()q.append(i)# 2.出队列# 单调队列的长度限制为b, 当队列元素个数大于b就需要将队首元素出队if i - q[0] + 1 >= b + 1:q.popleft()# 3.开始记录答案if i >= b - 1:# 单调队列降序, 队首元素为区间最小值w.append(nums[q[0]])ws.append(w)# 记录所有二维区间长度为a宽度为b的区间最小值ans = []# 将ws中的每一列作为一行tmps = [[ws[j][i] for j in range(n)] for i in range(len(ws[0]))]for i in range(len(ws[0])):nums = tmps[i]# w记录当前行所有区间长度为a的区间最小值w = []# 单调队列q = deque()for i, v in enumerate(nums):# 1.入队列while q and v <= nums[q[-1]]:q.pop()q.append(i)# 2.出队列# 单调队列的长度限制为a, 当队列元素个数大于a就需要将队首元素出队if i - q[0] + 1 >= a + 1:q.popleft()# 3.开始记录答案if i >= a - 1:# 单调队列降序, 队首元素为区间最小值w.append(nums[q[0]])ans.append(w)return ans
maxs = max_value_queue(datas)
mins = min_value_queue(datas)answer = 0
for i in range(len(maxs)):for j in range(len(maxs[0])):answer = (answer + maxs[i][j] * mins[i][j]) % 998244353
print(answer)

解题思路:

这个就是二维单调队列的使用, 一维单调队列求区间最值比较简单, 可以去做一下滑动窗口最大值:

https://leetcode.cn/problems/sliding-window-maximum/, 那么二维区间最值怎么求呢?

首先从横向来看,可以用单调队列将每一行的区间最值求出来, 用数组w记录当前行的所有区间最值, 用ws将每一行的w都存储一起。接下来再来纵看,取已经求得的ws二维数组其中一列,求纵向的区间最值, 用t记录当前列的区间最值, 用ts记录每一列的t。最后ts就存储了所有的子矩阵的区间最值。

将区间最大值和区间最小值相乘求和求模就是最终答案, 代码看起来很长, 其实功能比较相似, 两个函数都是求区间最值, 一个是最大一个是最小。

http://www.lryc.cn/news/325689.html

相关文章:

  • Nginx 故障排查之斜杠(/) --(附 Nginx 常用命令)
  • 【超全详解一文搞懂】Scala基础
  • 16:00面试,16:06就出来了,问的问题有点变态。。。
  • 【CTFshow 】web 通关 1.0
  • babel起手式
  • AI大模型在医疗领域的应用案例:自然语言处理与医疗文本分析
  • c语言常见错误
  • 分别使用TCP/UDP实现互相实时发送消息,接收消息功能
  • 使用阿里CICD流水线打包Vue项目到阿里的docker镜像私仓,并自动部署到服务器启动服务
  • 第十三届蓝桥杯物联网试题(省赛)
  • 将谷歌 Gemma AI大模型 部署安装本地教程(可离线使用)
  • ChatGPT提示词大全:解锁AI对话
  • rust中字符串String常用方法和注意事项
  • C语言:自定义类型(结构体)
  • 唯众物联网安装调试员实训平台物联网一体化教学实训室项目交付山东技师学院
  • SqlServer期末复习(数据库原理及应用)持续更新中
  • 合辑下载 | MatrixOne 与 MySQL 全面对比
  • Ubuntu 22.04安装Python3.10.13
  • 2.4 如何运行Python程序
  • Vue中如何实现动态改变字体大小
  • Spring实例化Bean的三种方式
  • AI研报:从Sora看多模态大模型发展
  • Unity访问安卓(Android)或苹果(iOS)相册
  • 用webpack 构建自己的vue-cli
  • ZCC6982最大充电电流 2A、升压型 2 节锂电池充电管理器
  • 【机器学习】无监督学习算法之:K均值聚类
  • 为wordpress特定分类目录下的内容添加自定义字段
  • javaWeb在线考试系统
  • 项目管理商业文件--商业论证与效益管理计划
  • 机器学习揭秘:解锁从理论到实践的每一步!