学习笔记 悬线法

  • 最大子矩阵问题:

在一个给定的矩形中有一些障碍点,找出内部不包含障碍点的,边与整个矩形平行或重合的最大子矩形。

极大子矩型:无法再向外拓展的有效子矩形

最大子矩型:最大的一个有效子矩形

特别的,在一个有障碍点的矩形中,最大子矩形一定是极大子矩形

  • 悬线法

悬线:上端覆盖了一个障碍点或者到达整个矩形上边界的有效线段

每个悬线上的点的与底部的点一一对应,矩形中每一个点(矩形顶部点除外)都对应了一条悬线。

如果把一条悬线向左右两个方向尽可能的移动,那么就得到了一个矩形。

注意:悬线对应的矩型不一定是极大子矩阵,因为悬线定义中固定了悬线的下边界,故而,悬线左右移动所得到的矩形无法向下扩展。

  • 悬线法的实现

      三个重要的元素:

  1. heighti,j :表示以( i ,j )为底的悬线的高 (初始化为1)
  2. lefti,j        :表示向左最多能移动到的位置  (初始化为j)
  3. righti,j     : 表示向右最多能移动到的位置   (初始化为j)

其中的left,right要视题目要求进行进一步的初始化,如例题

转移:

如果点  不是障碍点,那么,以  为底的悬线就等于以  为底的悬线加点  到点       的线段。因此,  。

当然还要注意左右边界的问题

以上图片转自  https://zhuanlan.zhihu.com/p/46382722

画个图理解一下

那么计算面积就轻而易举

对于以点  为底的悬线对应的子矩形,其面积计算为

问题解:

时间复杂度:  ;空间复杂度: 

例题:luogu P1169 棋盘制作

直接上代码

 #include<cstdio>
#include<algorithm>
#include<iostream>
#define maxn 2010
#define re register
using namespace std;
int n,m,ans1,ans2;
int map[maxn][maxn],height[maxn][maxn];
int l[maxn][maxn],r[maxn][maxn];
int main()
{
scanf("%d%d",&n,&m);
for(re int i=;i<=n;++i)
for(re int j=;j<=m;++j)
{
scanf("%d",&map[i][j]);
height[i][j]=;
l[i][j]=r[i][j]=j;
}
for(re int i=;i<=n;++i)
for(re int j=;j<=m;++j)
{
if(map[i][j]!=map[i][j-])
l[i][j]=l[i][j-];
}
for(re int i=;i<=n;++i)
for(re int j=m-;j>=;j--)
{
if(map[i][j]!=map[i][j+])
r[i][j]=r[i][j+];
}
//以上为初始化
for(re int i=;i<=n;++i)
for(re int j=;j<=m;++j)
{
if(i>&&map[i][j]!=map[i-][j])
{
height[i][j]=height[i-][j]+;
l[i][j]=max(l[i][j],l[i-][j]);
r[i][j]=min(r[i][j],r[i-][j]);
}
int a=r[i][j]-l[i][j]+;
int b=min(height[i][j],a);
ans1=max(ans1,a*height[i][j]);//最大矩形
ans2=max(ans2,b*b);//最大正方形
}
printf("%d\n%d",ans2,ans1);
return ;
}

注:部分内容转载自  
Flavius Buffon:悬线法用来求解最大子矩形问题 同时也是参考文献

[P1169] 棋盘制作 &悬线法学习笔记的更多相关文章

  1. 洛谷P1169 棋盘制作(悬线法)

    题目链接:https://www.luogu.org/problemnew/show/P1169 #include<bits/stdc++.h> #define fi first #def ...

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

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

  3. 洛谷P1169 [ZJOI2007]棋盘制作 悬线法 动态规划

    P1169 [ZJOI2007]棋盘制作 (逼着自己做DP 题意: 给定一个包含0,1的矩阵,求出一个面积最大的正方形矩阵和长方形矩阵,要求矩阵中相邻两个的值不同. 思路: 悬线法. 用途: 解决给定 ...

  4. P1169 [ZJOI2007]棋盘制作[悬线法/二维dp]

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

  5. P1169 [ZJOI2007]棋盘制作——悬线法

    ---恢复内容开始--- 给你一个矩阵,选出最大的棋盘,棋盘的要求是黑白相间(01不能相邻),求出最大的正方形和矩形棋盘的面积: 数据n,m<=2000; 这个一看就可能是n2DP,但是写不出. ...

  6. P1169 [ZJOI2007]棋盘制作 悬线法or单调栈

    思路:悬线法\(or\)单调栈 提交:2次 错因:正方形面积取错了\(QwQ\) 题解: 悬线法 讲解:王知昆\(dalao\)的\(PPT\) 详见代码: #include<cstdio> ...

  7. 【BZOJ-3039&1057】玉蟾宫&棋盘制作 悬线法

    3039: 玉蟾宫 Time Limit: 2 Sec  Memory Limit: 128 MBSubmit: 753  Solved: 444[Submit][Status][Discuss] D ...

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

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

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

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

随机推荐

  1. 初学Java Web(8)——过滤器和监听器

    什么是过滤器 过滤器就是 Servlet 的高级特性之一,就是一个具有拦截/过滤功能的一个东西,在生活中过滤器可以是香烟滤嘴,滤纸,净水器,空气净化器等,在 Web 中仅仅是一个实现了 Filter ...

  2. RabbitMQ消息队列(十)-高可用集群部署实战

    前几章讲到RabbitMQ单主机模式的搭建和使用,我们在实际生产环境中出于对性能还有可用性的考虑会采用集群的模式来部署RabbitMQ. RabbitMQ集群基本概念 Rabbit模式大概分为以下三种 ...

  3. SpringMvc 请求中日期类型参数接收一二事儿

    首先说明:以版本为Spring 4.3.0为测试对象: 开启<mvc:annotation-driven /> 测试场景一:请求中含有date属性,该类型为日期类型,SpringMvc采用 ...

  4. Autorelease机制讲解

    Autorelease机制是在iOS内存管理中的一员.在MRC中,是通过调用[obj autorelease]来延迟内存释放:在ARC中,我们已经完全不需要知道Autorelease就能很好地管理好内 ...

  5. Asp.Net路由重写为用户名或者ID

    有一个需求如下:指定某个Area的路由(Area:Wx)在其后面添加用户名或者ID作为URL参数,即像下面的样子: /Wx/xiaoming/ /Wx/xiaoming/photo /Wx/xiaom ...

  6. vb.net MakeWParam

    Private Function MakeWParam(loWord As Integer, hiWord As Integer) As Integer ) End Function

  7. Electron 创建一个空白的界面

    添加应用 首先添加一个Lorikeet版本的Electron应用. 'use strict' const electron = require('electron'); const app = ele ...

  8. Java开发笔记(三十九)日期工具Date

    Date是Java最早的日期工具,编程中经常通过它来获取系统的当前时间.当然使用Date也很简单,只要一个new关键字就能创建日期实例,就像以下代码示范的那样: // 创建一个新的日期实例,默认保存的 ...

  9. Kotlin入门学习笔记

    前言 本文适合人群 有一定的java基础 变量与方法 变量声明及赋值 var 变量名: 变量类型 val 变量名: 变量类型 这里,var表示可以改变的变量,val则是不可改变的变量(第一个赋值之后, ...

  10. vue项目中vscode格式化配置和eslint配置冲突

    问题描述 使用vscode开发vue项目的时候,从远端拉下一个新的项目后,安装完依赖后跑起项目时,发现直接报了一堆语法错误:包括换行.空格.单双引号.分号等各种格式问题 因为我的 vscode 安装使 ...