把悬线法这个坑填了,还是很简单的 qwq。

悬线法一般解决有一定约束条件的最大矩形问题。悬线的定义是从一个点一直往上走直到边界或者不符合条件结束。

炒个例子,在这题里面比如说有这样一个矩形

0 1 0 1
0 1 0 1
1 0 1 0
1 1 1 1

则第 \(4\) 行第 \(3\) 列的点的悬线就是这个

1
0
1(这就是起始的点

我们定义 \(up_{i, j}\) 为 \(i\) 行 \(j\) 列的点的悬线长度。

显而易见,如果可以转移,则有 \(up_{i, j} = up_{i - 1, j} + 1\)。


由于是矩形,我们还要枚举悬线往左移动能到的极远处于往右移动的极远处。

令 \(le_{i, j}\) 为第 \(i\) 行 \(j\) 列的点的悬线往左移动的极远处,\(ri_{i, j}\) 是往右的极远处。

如果 \(a_{i, j} \not = a_{i - 1, j}\),则有 \(le_{i, j} = \max\{le_{i, j}, le_{i - 1, j}\}, ri_{i, j} = \min\{ri_{i, j}, ri_{i - 1, j}\}\)。

为什么左边界取 \(\max\),右边界取 \(\min\) 呢?这不是求最近的边界吗?

其实这很好理解,首先我们画个图(丑陋预警(

拿这玩意儿画网格图我是什么个天才(悲

点 F 是我们当前所在的点,,我们假设 F 在第 \(i\) 行 \(j\) 列。EF 是 F 的悬线。CD 代表 EF 目前往左移动更新的极远处,GH 代表 EF 目前往右移动更新的极远处,AB 代表 \(i - 1\) 行 \(j\) 列的悬线往左的极远处,IJ 代表往右的极远处。

显然,我们的 CD 应该被更新成 AB,GH 应该被更新成 IJ。原因很简单,以 AB 为例,AB 已经被更新过了,是极左的,那么它为什么没有更新到 CD 那一行呢,因为 CD 与 AB 之间出现了不合法的,那么我们就不能更新到 AB,得更新到 CD。GH 与 IJ 的更新同理。

那么问题又来了,这能保证是极远处吗?显然可以。证明和上面差不多,不写了。

还有一个问题,我发现基本上所有的文章都没有提到,只是在代码里面写了怎么可以这样(恼,还有一个问题,还是上面那张图

图逐渐走向艺术派

我们会发现我们的方程没有考虑这里的红色部分是否合法。所以我们要在之前提前处理。如果可以 \(a_{i, j} \not = a_{i, j - 1}\),则 \(le_{i, j} = le_{i, j - 1}\)(这里 \(j\) 正序枚举)。如果 \(a_{i, j} \not = a_{i, j + 1}\) 则 \(ri_{i, j} = ri_{i, j + 1}\)(这里 \(j\) 倒序枚举)

这个应该很好理解因为我语文不好其实是懒我就不多解释了 qwq

代码:

//SIXIANG
#include <iostream>
#define MAXN 2000
#define QWQ cout << "QWQ" << endl;
using namespace std;
int a[MAXN + 10][MAXN + 10], up[MAXN + 10][MAXN + 10], le[MAXN + 10][MAXN + 10], ri[MAXN + 10][MAXN + 10];
int main() {
int n, m;
cin >> n >> m;
for(int p = 1; p <= n; p++)
for(int i = 1; i <= m; i++) {
cin >> a[p][i];
up[p][i] = 1;
le[p][i] = ri[p][i] = i;
} for(int p = 1; p <= n; p++)
for(int i = 2; i <= m; i++)
if(a[p][i] != a[p][i - 1])
le[p][i] = le[p][i - 1]; for(int p = 1; p <= n; p++)
for(int i = m - 1; i >= 1; i--)
if(a[p][i] != a[p][i + 1])
ri[p][i] = ri[p][i + 1]; int ans1 = 0, ans2 = 0;
for(int p = 1; p <= n; p++) {
for(int i = 1; i <= m; i++) {
if(p > 1 && a[p][i] != a[p - 1][i]) {
up[p][i] = up[p - 1][i] + 1;
le[p][i] = max(le[p][i], le[p - 1][i]);
ri[p][i] = min(ri[p][i], ri[p - 1][i]);
}
int a = up[p][i], b = ri[p][i] - le[p][i] + 1;
int c = min(a, b);
ans1 = max(ans1, c * c);
ans2 = max(ans2, a * b);
}
}
cout << ans1 << ' ' << ans2 << endl;
}

其它题目就可以用类似的板子套上去(所以我就理直气壮的不写啦(

题解 [ZJOI2007]棋盘制作的更多相关文章

  1. [洛谷P1169][题解][ZJOI2007]棋盘制作

    我不是题目的说 这道题运用了一种很巧妙的DP方式:悬线法 如图,蓝色为悬线,黄色为向两边延伸的长度 那么显然,最大子矩形的宽一定是这些黄线中最小的(证明从略) 所以我们可以维护三个数组: Up[i][ ...

  2. 悬线法 || BZOJ 1057: [ZJOI2007]棋盘制作 || Luogu P1169 [ZJOI2007]棋盘制作

    题面:P1169 [ZJOI2007]棋盘制作 题解: 基本是悬线法板子,只是建图判断时有一点点不同. 代码: #include<cstdio> #include<cstring&g ...

  3. BZOJ 1057: [ZJOI2007]棋盘制作 悬线法求最大子矩阵+dp

    1057: [ZJOI2007]棋盘制作 Description 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的黑 ...

  4. 洛谷 P1169 [ZJOI2007]棋盘制作

    2016-05-31 14:56:17 题目链接: 洛谷 P1169 [ZJOI2007]棋盘制作 题目大意: 给定一块矩形,求出满足棋盘式黑白间隔的最大矩形大小和最大正方形大小 解法: 神犇王知昆的 ...

  5. BZOJ1057 [ZJOI2007]棋盘制作(极大化思想)

    1057: [ZJOI2007]棋盘制作 Time Limit: 20 Sec  Memory Limit: 162 MB Submit: 1848  Solved: 936 [Submit][Sta ...

  6. bzoj 1057: [ZJOI2007]棋盘制作 单调栈

    题目链接 1057: [ZJOI2007]棋盘制作 Time Limit: 20 Sec  Memory Limit: 162 MBSubmit: 2027  Solved: 1019[Submit] ...

  7. BZOJ 1057: [ZJOI2007]棋盘制作( dp + 悬线法 )

    对于第一问, 简单的dp. f(i, j)表示以(i, j)为左上角的最大正方形, f(i, j) = min( f(i + 1, j), f(i, j + 1), f(i + 1, j + 1)) ...

  8. P1169 [ZJOI2007]棋盘制作 && 悬线法

    P1169 [ZJOI2007]棋盘制作 给出一个 \(N * M\) 的 \(01\) 矩阵, 求最大的正方形和最大的矩形交错子矩阵 \(n , m \leq 2000\) 悬线法 悬线法可以求出给 ...

  9. [luogu P1169] [ZJOI2007]棋盘制作

    [luogu P1169] [ZJOI2007]棋盘制作 题目描述 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8*8大小的 ...

  10. 1057: [ZJOI2007]棋盘制作

    1057: [ZJOI2007]棋盘制作 https://www.lydsy.com/JudgeOnline/problem.php?id=1057 分析: 首先对于(i+j)&1的位置0-& ...

随机推荐

  1. 打印菱形-java

    public class WeekendDemo01 { /** 打印菱形 * * * *** * ***** * *** * * */ public static void main(String[ ...

  2. 【实时数仓】Day01-数据采集层:数仓分层、实时需求、架构分析、日志数据采集(采集到指定topic和落盘)、业务数据采集(MySQL-kafka)、Nginx反向代理、Maxwell、Canel

    一.数仓分层介绍 1.实时计算与实时数仓 实时计算实时性高,但无中间结果,导致复用性差 实时数仓基于数据仓库,对数据处理规划.分层,目的是提高数据的复用性 2.电商数仓的分层 ODS:原始日志数据和业 ...

  3. python中函数教程

    函数的基本概念 1.什么是函数? 函数相当于一种工具,就是把一串代码装到一起,我们下次需要用的这个功能的时候可以直接使用 函数相当于是工具(具有一定功能) 不用函数 修理工需要修理器件要用锤子 原地打 ...

  4. 第一百一十七篇: JavaScript 工厂模式和原型模式

    好家伙,本篇为<JS高级程序设计>第八章"对象.类与面向对象编程"学习笔记   1.工厂模式 工厂模式是另外一种关注对象创建概念的创建模式. 它的领域中同其它模式的不同 ...

  5. Vue3 企业级优雅实战 - 组件库框架 - 9 实现组件库 cli - 上

    上文搭建了组件库 cli 的基础架子,实现了创建组件时的用户交互,但遗留了 cli/src/command/create-component.ts 中的 createNewComponent 函数,该 ...

  6. gRPC入门与实操(.NET篇)

    为什么选择 gRPC 历史 长久以来,我们在前后端交互时使用WebApi + JSON方式,后端服务之间调用同样如此(或者更久远之前的WCF + XML方式).WebApi + JSON 是优选的,很 ...

  7. 写一个 Markdown 博客客户端

    这个"伪需求"是最近才想到的. 关于文章管理的想法,说来话长.我最初是在 CSDN 写技术文章,就用网页上的编辑器.后来在 CppBlog 写,用上了 Windows Live W ...

  8. JavaScript 图像压缩

    JavaScript 可以使用类似于 canvas 和 web workers 来实现图像压缩. 使用 canvas,可以将图像绘制到 canvas 上,然后使用 canvas 提供的 toBlob( ...

  9. 激光炸弹【算法竞赛进阶指南, HNOI2003】

    激光炸弹 地图上有 \(N\) 个目标,用整数 \(Xi,Yi\)表示目标在地图上的位置,每个目标都有一个价值 \(Wi\). 注意:不同目标可能在同一位置. 现在有一种新型的激光炸弹,可以摧毁一个包 ...

  10. 音乐解锁工具v1.10.3,QQ音乐,网易云,酷狗音乐格式转换工具,ncm转mp3,kgm转mp3,kgma转mp3,mgg转mp3,mflac转mp3,qmc转mp3,xm转mp3,kwm转mp3

    现在主流的听歌软件开会员听歌和下载单曲已经是两套业务了,即使开了会员下载的单曲也只能用固定的播放器进行播放,不能使用其他软件播放. 目前QQ音乐.网抑云音乐.酷狗等会员歌曲下载后都不再是传统的音频文件 ...