悬线法DP总结

问题模型

求满足某种条件(如01交替)的最大矩形(正方形)

思想

先预处理出\(ml[i][j],mr[i][j],mt[i][j]\),分别表示当前位置\((i,j)\)能向左扩展到的最左边的编号、能向右扩展到的最右边的编号、能向上扩展到的最大高度。

然后在做\(DP\)时,除第一行,每行根据上一行的状态更新当前状态,逐行扫一遍。复杂度\(O(n\times m)\)

[USACO5.3]巨大的牛棚Big Barn

洛谷题面

题意:求最大正方形

#include <cstdio>
#define MAXN 1010
#define MAX(A,B) ((A)>(B)?(A):(B))
#define MIN(A,B) ((A)<(B)?(A):(B))
using namespace std;
int n,t,ans;
bool mp[MAXN][MAXN];
int ml[MAXN][MAXN],mr[MAXN][MAXN],mt[MAXN][MAXN];
int main()
{
scanf("%d %d", &n, &t);
while(t--){
int x,y;
scanf("%d %d", &x, &y);
mp[x][y]=1;
}
for(register int i=1;i<=n;++i){
for(register int j=1;j<=n;++j){
ml[i][j]=mr[i][j]=j;
mt[i][j]=1;
}
}
for(register int i=1;i<=n;++i)
for(register int j=2;j<=n;++j)
if(mp[i][j-1]==0&&mp[i][j]==0)
ml[i][j]=ml[i][j-1]; // 预处理出能向左扩展到的最左边的编号
for(register int i=1;i<=n;++i)
for(register int j=n-1;j>=1;--j)
if(mp[i][j+1]==0&&mp[i][j]==0)
mr[i][j]=mr[i][j+1]; // 预处理出能向右扩展到的最右边的编号
for(register int i=1;i<=n;++i)
for(register int j=2;j<=n;++j){
if(i>1&&mp[i-1][j]==0&&mp[i][j]==0){ // 除第一行,每行根据上一行的状态更新当前状态
ml[i][j]=MAX(ml[i][j], ml[i-1][j]); // 注意取MAX,获得合法的ml
mr[i][j]=MIN(mr[i][j], mr[i-1][j]); // 注意取MIN,获得合法的mr
mt[i][j]=mt[i-1][j]+1;
}
int w=MIN(mr[i][j]-ml[i][j]+1, mt[i][j]); // 因为是正方形
ans=MAX(ans, w);
}
printf("%d", ans);
return 0;
}

[ZJOI2007]棋盘制作

P1169 洛谷题面

题意:求最大01交替正方形、长方形

#include <cstdio>
#define MAXN 2002
#define INF 0x3fffffff
#define MAX(A,B) ((A)>(B)?(A):(B))
#define MIN(A,B) ((A)<(B)?(A):(B))
using namespace std;
int n,m,ml[MAXN][MAXN],mr[MAXN][MAXN],mt[MAXN][MAXN];
int ans1, ans2;
bool mp[MAXN][MAXN];
int main()
{
scanf("%d %d", &n, &m);
for(int i=1;i<=n;++i)
for(int j=1;j<=m;++j)
scanf("%d", &mp[i][j]),ml[i][j]=mr[i][j]=j,mt[i][j]=1;
for(int i=1;i<=n;++i)
for(int j=2;j<=m;++j)
if(mp[i][j]!=mp[i][j-1]) // 01扩展
ml[i][j]=MIN(ml[i][j-1], ml[i][j]); // 预处理出能向左扩展到的最左边的编号
for(int i=1;i<=n;++i)
for(int j=m-1;j>=1;--j)
if(mp[i][j]!=mp[i][j+1]) // 01扩展
mr[i][j]=MAX(mr[i][j+1], mr[i][j]); // 预处理出能向右扩展到的最右边的编号
for(int i=1;i<=n;++i)
for(int j=2;j<=m;++j){
if(i>1&&mp[i][j]!=mp[i-1][j]){ // 除第一行,每行根据上一行的状态更新当前状态
ml[i][j]=MAX(ml[i][j], ml[i-1][j]);
mr[i][j]=MIN(mr[i][j], mr[i-1][j]);
mt[i][j]=mt[i-1][j]+1;
}
int w = mr[i][j] - ml[i][j]+1;
int h = mt[i][j];
ans1 = MAX(MIN(w,h)*MIN(w,h), ans1); // 找出最大合法正方形
ans2 = MAX(w*h, ans2); // 找出最大合法矩形
}
printf("%d\n%d", ans1, ans2);
return 0;
}

P4147 玉蟾宫

P4147 玉蟾宫

题意:毒瘤输入,求最大正方形

#include <cstdio>
#define MAXN 1010
#define MAX(A,B) ((A)>(B)?(A):(B))
#define MIN(A,B) ((A)<(B)?(A):(B))
using namespace std;
int n,m,ans;
bool mp[MAXN][MAXN];
int ml[MAXN][MAXN],mr[MAXN][MAXN],mt[MAXN][MAXN];
int read(){
char in=getchar();
while(in!='F'&&in!='R') in=getchar();
if(in=='F') return 1;
return 0;
}
int main()
{
scanf("%d %d", &n, &m);
for(register int i=1;i<=n;++i){
for(register int j=1;j<=m;++j){
mp[i][j]=read();
ml[i][j]=mr[i][j]=j;
mt[i][j]=1;
}
}
for(register int i=1;i<=n;++i)
for(register int j=2;j<=m;++j)
if(mp[i][j-1]==1&&mp[i][j]==1)
ml[i][j]=ml[i][j-1];
for(register int i=1;i<=n;++i)
for(register int j=m-1;j>=1;--j)
if(mp[i][j+1]==1&&mp[i][j]==1)
mr[i][j]=mr[i][j+1];
for(register int i=1;i<=n;++i)
for(register int j=2;j<=m;++j){
if(i>1&&mp[i-1][j]==1&&mp[i][j]==1){
ml[i][j]=MAX(ml[i][j], ml[i-1][j]);
mr[i][j]=MIN(mr[i][j], mr[i-1][j]);
mt[i][j]=mt[i-1][j]+1;
}
int w=mr[i][j]-ml[i][j]+1;
ans=MAX(ans, w*mt[i][j]);
}
printf("%d", 3*ans);
return 0;
}

悬线法DP总结的更多相关文章

  1. [ZJOI2007]棋盘制作 悬线法dp 求限制下的最大子矩阵

    https://www.luogu.org/problemnew/show/P1169 第一次听说到这种dp的名称叫做悬线法,听起来好厉害 题意是求一个矩阵内的最大01交错子矩阵,开始想的是dp[20 ...

  2. 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)) ...

  3. P1169 [ZJOI2007]棋盘制作 DP悬线法

    题目描述 国际象棋是世界上最古老的博弈游戏之一,和中国的围棋.象棋以及日本的将棋同享盛名.据说国际象棋起源于易经的思想,棋盘是一个8 \times 88×8大小的黑白相间的方阵,对应八八六十四卦,黑白 ...

  4. P4147 玉蟾宫 二维DP 悬线法

    题目背景 有一天,小猫rainbow和freda来到了湘西张家界的天门山玉蟾宫,玉蟾宫宫主蓝兔盛情地款待了它们,并赐予它们一片土地. 题目描述 这片土地被分成N*M个格子,每个格子里写着'R'或者'F ...

  5. hdu4328(经典dp用悬线法求最大子矩形)

    http://wenku.baidu.com/view/728cd5126edb6f1aff001fbb.html 关于悬线法,这里面有详解. 我当时只想到了记录最大长度,却没有想到如果连最左边和最右 ...

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

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

  7. 悬线法——有套路的DP

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

  8. 【题解】洛谷P1169 [ZJOI2007] 棋盘制作(坐标DP+悬线法)

    次元传送门:洛谷P1169 思路 浙江省选果然不一般 用到一个从来没有听过的算法 悬线法: 所谓悬线法 就是用一条线(长度任意)在矩阵中判断这条线能到达的最左边和最右边及这条线的长度 即可得到这个矩阵 ...

  9. DP(悬线法)+二维前缀和【p2706】巧克力

    Background 王7的生日到了,他的弟弟准备送他巧克力. Description 有一个被分成n*m格的巧克力盒,在(i,j)的位置上有a[i,j]块巧克力.就在送出它的前一天晚上,有老鼠夜袭巧 ...

随机推荐

  1. Codeforces Round #570 Div. 3

    A:暴力从小到大枚举判断. #include<bits/stdc++.h> using namespace std; #define ll long long #define inf 10 ...

  2. 获取电脑 ip 地址 及系统

    public static void main(String[] args) throws UnknownHostException { //获取电脑系统 结果:os.name:Windows 10 ...

  3. swagger 报错:illegal defaultValue null for param type integer

    swagger(版本2.9.2) 刷新报错,错误信息如下图: 问题原因: 根据上面这句报错信息,点进去AbstractSerializableParameter.java:412可以看到 源码, @J ...

  4. Linux守护进程编写指南

    Linux守护进程编写指南 守护进程(Daemon)是运行在后台的一种特殊进程.它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件.守护进程是一种很有用的进 程.Linux的大多数服务器 ...

  5. DevExtreme学习笔记(一) DataGrid中MVC分析

    @(Html.DevExtreme().DataGrid() .ID("gridContainer") .DataSource(d => d .OData() .Url(&q ...

  6. Net文件递归查找并保存

    原理:遍历当前文件夹的子文件,保存遍历文件夹下的所有文件 主要方法(2个): //获取文件夹下的所有文件 并保存 string[] path = Directory.GetFiles(NeedFile ...

  7. CPU子系统

    CPU的基本结构: CPU的主要部件: ​ 运算部件.缓存部件.寄存器.控制器.时序部件 CPU的工作原理: ​ 主要功能:处理指令.执行操作.控制时间.数据运算 ​ 执行指令的流程:读取指令.指令译 ...

  8. javascript, 元素 页面 简单碰撞

     <div>     </div>     <style>         div {             width: 100px;              ...

  9. Vue项目中使用svg图标

    svg的图片有着颜色自定义大小自定义的优势.在阿里图标库可以下载svg图片. 那么在vue框架中我们该怎么使用svg图片呢 这个时候就用到了 webpack 插件 svg-sprite-loader ...

  10. Jmeter学习笔记(十二)——断言

    一.断言简介 jmeter中有个元件叫做断言(Assertion).用于检查测试中得到的响应数据等是否符合预期,用以保证性能测试过程中的数据交互与预期一致. 使用断言的目的:在request的返回层面 ...