悬线法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. org.apache.ibatis.binding.BindingException: Invalid bound statement (not found): 错误解决

    报错信息:org.apache.ibatis.binding.BindingException: Invalid bound statement (not found) 说明:这段报错信息表示 Map ...

  2. .Net Core 注入学习——注册服务

    解析 .Net Core 注入——注册服务发表于:2017-10-23 10:47 作者:行走即歌 来源:51Testing软件测试网采编字体:大 中 小 | 上一篇 | 下一篇 |我要投稿 | 推荐 ...

  3. C# vb .net实现透视反射效果

    在.net中,如何简单快捷地实现Photoshop滤镜组中的透视反射效果呢?答案是调用SharpImage!专业图像特效滤镜和合成类库.下面开始演示关键代码,您也可以在文末下载全部源码: 设置授权 第 ...

  4. MVC4 部署 could not load file or assembly system.web.http.webhost 或是其它文件出误

    自从VS2010发布之后使用它来做开发的程序员越来越多,其中很多人使用了MVC来作为新的开发框架,但是在系统部署的时候我们也遇到诸多问题,因为目前大多数windows服务器采用的还是Windows S ...

  5. Git使用整理

    [本文由水木桶首发于博客园,原文地址:https://www.cnblogs.com/shuimutong/p/11404664.html,未接允许,严禁转载] 背景 很久之前使用的是svn,直接在E ...

  6. node+mysql+vue+express项目搭建

    第一步:项目搭建之前首先需要安装node环境和MySQL数据库. 在已经完成上述的条件下开始进行以下操作: npm install @vue/cli -g   (-g 代表全局安装) 初始化项目  v ...

  7. JavaScript对象复制

    近期项目因为怕数据污染所以用到了js的对象复制 js里的对象都是继承自object,是引用类型,所以无法通过=号复制 所以整理了一些常用的复制方法,如下 一.通过JSON序列化和反序列化创建新的对象 ...

  8. React Native 开发豆瓣评分(八)首页开发

    首页完成效果展示: 一.开发占位图组件 在没有数据的时候使用占位图替代 items 的位置. 在 components 目录里创建 moviesItemPlaceholder.js import Re ...

  9. JavaScript之获取标签

    (1)html <p id="txt">这是P标签</p> <h1 id="txt" style="color: red ...

  10. iPhone的xib与iPad的xib相互转换

    1. xib转换 iPhone版本APP开发完成后需要再开发iPad版本的APP,需要把iPhone版本的xib文件添加到iPad项目中去,但是Xcode中iPhone和iPad使用的xib格式不能完 ...