luogu1169 棋盘制作
题目大意
有一个有m*n个格子的矩形,每个格子都有黑或白两种颜色。现要求将该矩形分别裁剪成一个小矩形或一个小正方形,使得这个矩形和正方形是个国际象棋棋盘,且面积最大。
题解
首先,为了简化问题,我们每隔一个格子将颜色翻转,这样题目就变成了最大01子矩阵问题。解决这类问题需要用到悬线法。
首先,我们规定一个半极大子矩阵为上、左、右边缘由于有障碍而不可继续向上、左、右方向延伸的矩阵。显然我们要求的矩阵是一个半极大子矩阵。对于一个这样的矩阵,我们定义悬线为上边缘的上一排的一个障碍点所在竖直线在矩阵内的部分。那么一个矩阵就相当于悬线在没有障碍的范围内左右移动的结果。因为矩阵中的每一个点都只对应唯一一个悬线,所以我们可以用通过枚举矩形中的每一个点的方法来枚举到所有的半极大子矩阵。首先悬线的长度可以递推求解。定义一个点对应悬线在半极大子矩形中向右移动的长度为RecR[row][col],向左移动的长度为RecL[row][col]。要知道这两个量,我们还需要知道该点可向左最远延伸的距离L[row][col]和向右最远延伸的距离R[row][col],对于一个点,其RecL[row][col] = min(row1,col1与row,col在同一悬线内,包括row, col){L[row1][col1]}。这样子矩阵的面积就可求了。
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std; #define Clear(a, val) memset(a, val, sizeof(a))
const int MAX_ROW = 2010, MAX_COL = 2010, INF = 0x3f3f3f3f;
int A[MAX_ROW][MAX_COL];
int HorL[MAX_ROW][MAX_COL], HorR[MAX_ROW][MAX_COL], H[MAX_ROW][MAX_COL], RecL[MAX_ROW][MAX_COL], RecR[MAX_ROW][MAX_COL];//Hor:Horizontal Rec:Rectangle
int TotRow, TotCol, AnsSq, AnsRec; void ChangeMap()//一黑一白改为连续黑连续白
{
int st = 1;
for (int row = 1; row <= TotRow; row++)
{
for (int col = st; col <= TotCol; col += 2)
A[row][col] = !A[row][col];
st = st == 1 ? 2 : 1;
}
} void Solve()
{
Clear(HorL, 0), Clear(HorR, 0), Clear(H, 0), Clear(RecL, INF), Clear(RecR, INF);
for (int row = 1; row <= TotRow; row++)
{
for (int col = 1; col <= TotCol; col++)
{
if(A[row][col])
{
HorL[row][col] = HorL[row][col - 1] + 1;
H[row][col] = H[row - 1][col] + 1;
}
}
for (int col = TotCol; col >= 1; col--)
HorR[row][col] = A[row][col] ? HorR[row][col + 1] + 1 : 0;
} for (int row = 1; row <= TotRow; row++)
{
for (int col = 1; col <= TotCol; col++)
{
if (A[row][col])
{
RecL[row][col] = min(HorL[row][col], RecL[row - 1][col]);
RecR[row][col] = min(HorR[row][col], RecR[row - 1][col]);
}
else
RecL[row][col] = RecR[row][col] = INF;
int curSqEdge = min(H[row][col], RecL[row][col] + RecR[row][col] - 1);
AnsSq = max(AnsSq, curSqEdge * curSqEdge);
AnsRec = max(AnsRec, H[row][col] * (RecR[row][col] + RecL[row][col] - 1));
}
}
} void ReverseMap()
{
for (int row = 1; row <= TotRow; row++)
for (int col = 1; col <= TotCol; col++)
A[row][col] = !A[row][col];
} int main()
{
scanf("%d%d", &TotRow, &TotCol);
for (int row = 1; row <= TotRow; row++)
for (int col = 1; col <= TotCol; col++)
scanf("%d", &A[row][col]);
ChangeMap();
Solve();
ReverseMap();
Solve();
printf("%d\n%d\n", AnsSq, AnsRec);
return 0;
}
luogu1169 棋盘制作的更多相关文章
- luogu1169 棋盘制作 (单调栈)
先预处理出来从每个位置 以0开始 往右交替最多能放多少格 然后就相当于对每一列做HISTOGRA #include<bits/stdc++.h> #define pa pair<in ...
- 【BZOJ-3039&1057】玉蟾宫&棋盘制作 悬线法
3039: 玉蟾宫 Time Limit: 2 Sec Memory Limit: 128 MBSubmit: 753 Solved: 444[Submit][Status][Discuss] D ...
- 洛谷 P1169 [ZJOI2007]棋盘制作
2016-05-31 14:56:17 题目链接: 洛谷 P1169 [ZJOI2007]棋盘制作 题目大意: 给定一块矩形,求出满足棋盘式黑白间隔的最大矩形大小和最大正方形大小 解法: 神犇王知昆的 ...
- BZOJ1057 [ZJOI2007]棋盘制作(极大化思想)
1057: [ZJOI2007]棋盘制作 Time Limit: 20 Sec Memory Limit: 162 MB Submit: 1848 Solved: 936 [Submit][Sta ...
- bzoj 1057: [ZJOI2007]棋盘制作 单调栈
题目链接 1057: [ZJOI2007]棋盘制作 Time Limit: 20 Sec Memory Limit: 162 MBSubmit: 2027 Solved: 1019[Submit] ...
- 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)) ...
- [P1169] 棋盘制作 &悬线法学习笔记
学习笔记 悬线法 最大子矩阵问题: 在一个给定的矩形中有一些障碍点,找出内部不包含障碍点的,边与整个矩形平行或重合的最大子矩形. 极大子矩型:无法再向外拓展的有效子矩形 最大子矩型:最大的一个有效子矩 ...
- 悬线法 || BZOJ 1057: [ZJOI2007]棋盘制作 || Luogu P1169 [ZJOI2007]棋盘制作
题面:P1169 [ZJOI2007]棋盘制作 题解: 基本是悬线法板子,只是建图判断时有一点点不同. 代码: #include<cstdio> #include<cstring&g ...
- P1169 [ZJOI2007]棋盘制作 && 悬线法
P1169 [ZJOI2007]棋盘制作 给出一个 \(N * M\) 的 \(01\) 矩阵, 求最大的正方形和最大的矩形交错子矩阵 \(n , m \leq 2000\) 悬线法 悬线法可以求出给 ...
随机推荐
- JS——放大镜
放大镜: 1.比例系数要恒定:在系数为4的情况下,若原图是820*512,那么小图必须是205*128,放大镜若是50,原图显示区域必须200 2.计算鼠标在小图中的坐标 3.放大镜需要在鼠标中间位置 ...
- CSS——◇demo
核心思想:嵌套盒子中的◇超过父盒子的部分隐藏. 第一种写法: <!DOCTYPE html> <html> <head> <meta charset=&quo ...
- 【技术累积】【点】【java】【28】Map遍历
Map遍历 map的遍历一般有几种吧 遍历entrySet for(Map.Entry<String,String> entry : map.entrySet()){ } Iterator ...
- SpringMVC知识点总结一(非注解方式的处理器与映射器配置方法)
一.SpringMVC处理请求原理图(参见以前博客) 1. 用户发送请求至前端控制器DispatcherServlet 2. DispatcherServlet收到请求调用HandlerMappi ...
- 27.8 执行定时计算限制操作(Timer)
private static System.Threading.Timer s_Timer; static void Main() { Console.WriteLine("checking ...
- BFS入门篇——RQNOJ195&&335
PID195 / 校园迷宫☆ 从x,y走到去q,w的最小步数,限制是有的点可走,有的点不可走,BFS嘛. #include<bits/stdc++.h> using namespace s ...
- Linux:用户和组总结
从创建文件说起:useradd xiaomi 这里是创建了xiaomi用户 默认系统还会创建:/home/xiaomi /var/mail/xiaomi 即家目录和 ...
- Bootstrap练习:可折叠下拉菜单
代码: <!DOCTYPE html> <html lang="zh-cn"> <head> <meta charset="UT ...
- Python 查看关键字
关键字 import keyword print(keyword.kwlist)
- 渗透实战(周六):Hydra&Metasploit暴力破解SSH登录口令
一. SSH服务开启前基础配置 1.1 修改配置文件