Description

给定一个大小为 \(n\times m\) 的 \(01\) 矩阵。

要求支持:单点翻转,询问子矩形内部最大正方形。

\(n\times m\leq 4\cdot 10^6,n\leq m,q\leq 2000\)。

Sol

线段树神题。

我们来一步步解决问题。

首先考虑询问整个矩形,且只有一次询问怎么做。

我们可以 \(O(n)\) 的枚举矩形的上边界,再 \(O(m)\) 的枚举右边界,再安排一个指针表示矩形的左边界,这样可以发现,随着右边界增大,这个指针是单调不减的。那怎么判断当前枚举的正方形是否合法呢?根据这个单调不减的性质,我们可以用一个单调队列来维护这个左指针,具体就不细说了,大概就是每个点记录一个向下最长的一段 \(1\) 能够延伸多长就好了。

然后回到该问题。

观察到 \(q,m\) 比较小,所以可以让复杂度偏向 \(q,m\)。

对 \(n\) 这一维开一棵线段树,线段树上每个节点表示的矩形高度都为 \(m\)。

然后问题就是如何合并信息,即知道两个子区间的信息能否合并出大区间中跨越 \(mid\) 的最大子矩形。

所以我们除了在每个节点维护最大子矩形之外,还要维护两个信息 \(lmx[i],rmx[i]\) 表示第 \(i\) 行从左端点和右端点分别最长能延伸多长的一段 \(1\)。

然后就支持合并了,具体就是,枚举一个子矩形的下边界 \(i\),然后用两个单调队列维护左儿子的右边界和右儿子的左边界,再拿个指针维护子矩形的上边界,就可以均摊 \(O(m)\) 求出以每行为下边界跨越 \(mid\) 的最大子矩形了。

不想说了不想说了。

看代码吧。

Code

#pragma GCC optimize(2)
#include<bits/stdc++.h>
using namespace std;
typedef double db;
typedef long long ll;
const int N=4e6+5;
#define ls x<<1
#define rs x<<1|1
#define lss ls,l,mid,ql,qr
#define rss rs,mid+1,r,ql,qr int q1[N],q2[N];
int n,m,q,ret,len;
int hd1,tail1,hd2,tail2; struct Node{
int f[N<<2];
int* operator[](int x){return f+x*m;}//这个重载中括号很巧很巧
}mp,lmx,rmx,sum; void pushup(int x,int l,int r){
int mid=l+r>>1,len1=mid-l+1,len2=r-mid; //[l,mid] [mid+1,r]
hd1=hd2=1;tail1=tail2=0;
for(int j=1,i=1;i<=m;i++){
while(hd1<=tail1 and rmx[ls][q1[tail1]]>=rmx[ls][i]) tail1--; q1[++tail1]=i;
while(hd2<=tail2 and lmx[rs][q2[tail2]]>=lmx[rs][i]) tail2--; q2[++tail2]=i;
while(hd1<=tail1 and hd2<=tail2 and i-j+1>rmx[ls][q1[hd1]]+lmx[rs][q2[hd2]]){
j++;
if(q1[hd1]<j) hd1++;
if(q2[hd2]<j) hd2++;
}
sum[x][i]=max(sum[ls][i],sum[rs][i]);
if(hd1<=tail1 and hd2<=tail2) sum[x][i]=max(sum[x][i],i-j+1);
}
for(int i=1;i<=m;i++)
lmx[x][i]=lmx[ls][i]+(lmx[ls][i]==len1?lmx[rs][i]:0),
rmx[x][i]=rmx[rs][i]+(rmx[rs][i]==len2?rmx[ls][i]:0);
} void build(int x,int l,int r){
if(l==r){
for(int i=1;i<=m;i++)
lmx[x][i]=rmx[x][i]=sum[x][i]=mp[l][i];
return;
} int mid=l+r>>1;
build(ls,l,mid),build(rs,mid+1,r);
pushup(x,l,r);
} void modify(int x,int l,int r,int ql,int qr,int c){
if(l==r){
lmx[x][c]^=1;rmx[x][c]=sum[x][c]=lmx[x][c];
return;
} int mid=l+r>>1;
ql<=mid?modify(lss,c):modify(rss,c);
pushup(x,l,r);
} void merge(int x,int l,int r,int QL,int QR){
hd1=hd2=1;tail1=tail2=0;
int len1=len,len2=r-l+1;
for(int j=QL,i=QL;i<=QR;i++){
while(hd1<=tail1 and rmx[0][q1[tail1]]>=rmx[0][i]) tail1--; q1[++tail1]=i;
while(hd2<=tail2 and lmx[x][q2[tail2]]>=lmx[x][i]) tail2--; q2[++tail2]=i;
while(hd1<=tail1 and hd2<=tail2 and i-j+1>rmx[0][q1[hd1]]+lmx[x][q2[hd2]]){
j++;
if(q1[hd1]<j) hd1++;
if(q2[hd2]<j) hd2++;
}
if(hd1<=tail1 and hd2<=tail2) ret=max(ret,i-j+1);
}
for(int i=QL;i<=QR;i++)
lmx[0][i]=lmx[0][i]+(lmx[0][i]==len1?lmx[x][i]:0),
rmx[0][i]=rmx[x][i]+(rmx[x][i]==len2?rmx[0][i]:0);
} void query(int x,int l,int r,int ql,int qr,int QL,int QR){
if(ql<=l and r<=qr){
for(int i=QL;i<=QR;i++)
ret=max(ret,min(sum[x][i],i-QL+1));
merge(x,l,r,QL,QR);
len+=r-l+1;
return;
} int mid=l+r>>1;
if(ql<=mid) query(lss,QL,QR);
if(mid<qr) query(rss,QL,QR);
} int query(int ql,int qr,int QL,int QR){
ret=len=0;
for(int i=1;i<=m;i++) lmx[0][i]=rmx[0][i]=0;
query(1,1,n,ql,qr,QL,QR);
return ret;
} signed main(){
scanf("%d%d%d",&n,&m,&q);
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%d",&mp[i][j]);
build(1,1,n);
while(q--){
int opt; scanf("%d",&opt);
if(!opt){
int x,y; scanf("%d%d",&x,&y);
modify(1,1,n,x,x,y);
} else{
int l,s,r,t; scanf("%d%d%d%d",&l,&s,&r,&t);
printf("%d\n",query(l,r,s,t));
}
} return 0;
}

[Code+#3] 寻找车位的更多相关文章

  1. CODE[VS]-寻找子串位置-字符串处理-天梯青铜

    题目描述 Description 给出字符串a和字符串b,保证b是a的一个子串,请你输出b在a中第一次出现的位置. 输入描述 Input Description 仅一行包含两个字符串a和b 输出描述 ...

  2. loj #6302. 「CodePlus 2018 3 月赛」寻找车位【线段树+单调队列】

    考虑静态怎么做:枚举右边界,然后枚举上边界,对应的下边界一定单调不降,单调栈维护每一列从当前枚举的右边界向左最长空位的长度,这样是O(nm)的 注意到n>=m,所以m<=2000,可以枚举 ...

  3. [luogu4259]寻找车位

    考虑一个分治的做法:按行分治,将所有区间分为两类--经过分割线的.在左/右区间内部,后者显然可以递归下取,考虑前者 先求出出该行上每一列向上和向下的最大长度,记作$up_{i}$和$down_{i}$ ...

  4. <停车卫> 产品需求说明书 version 2.0

    <停车卫> 产品需求说明书 文档版本号: Version 2.0 文档编号: xxxx 文档密级: 归属部门/项目: 产品名: 停车卫 子系统名: 编写人: kina 编写日期: 2015 ...

  5. 瞬间读懂什么是互联网思维、大数据、O2O、众筹、红海

     1.什么叫大数据? 某必胜客店的电话铃响了,客服人员拿起电话. 客服:必胜客.您好,请问有什么需要我为您服务? 顾客:你好,我想要一份…… 客服:先生,烦请先把您的会员卡号告诉我. 顾客:16846 ...

  6. MySQL的诡异同步问题-重复执行一条relay-log

    MySQL的诡异同步问题 近期遇到一个诡异的MySQL同步问题,经过多方分析和定位后发现居然是由于备份引发的,非常的奇葩,特此记录一下整个问题的分析和定位过程. 现象 同事扩容的一台slave死活追不 ...

  7. 中文分词系列(一) 双数组Tire树(DART)详解

    1 双数组Tire树简介 双数组Tire树是Tire树的升级版,Tire取自英文Retrieval中的一部分,即检索树,又称作字典树或者键树.下面简单介绍一下Tire树. 1.1 Tire树 Trie ...

  8. 线下市场,选择微信小程序从未显得如此重要

    2017 年 1 月 9 日,小程序正式上线,到今日,3 月 8 号,这个新产品面世刚好满两个月.小程序刚推出便受到全球关注,腾讯股价当天即创逾一个月高位,但关注度先是急速上涨,不久便迅速降温,甚至在 ...

  9. Python(简单计算器)

    参考:https://www.cnblogs.com/alex3714/articles/5169958.html import re ret = re.search('\([^()]+\)','(1 ...

随机推荐

  1. [zt+总结]wpf 应用权限问题

    一.Inno Setup打包添加和去除管理员权限 转载:https://www.cnblogs.com/walker-lc/articles/3470679.html 添加管理员权限 1.在[Setu ...

  2. h5、css3基础

    一.html(超文本标记语言) 作用:实现页面布局 页面由许多标记符号组成 由浏览器解释执行 二.html主题创建方式 !(英文状态)+tab html:4s+tab html:5+tab 三.标签 ...

  3. async与defer

    <script>元素的几种常见属性: async  异步加载,立即下载,不应妨碍页面其他操作,标记为 async 的异步脚本并不保证按照指定的先后顺序执行,因此异步脚本不应该在加载期间修改 ...

  4. c# 集合的长度为什么是可变的

    摘要: 写在前面:此随笔仅仅是作为个人学习总结,有不对的地方,请各位前辈指正O(∩_∩)O........ 一: 引入 在学习集合之前我们都学习过数组.可以知道数组的长度在声明的时候就已经被固定了,不 ...

  5. 做个流量站-聚茶吧, 汇聚"茶"的地方

    犹豫了好久,终于下定决心,做一回个人站长了,虽然没啥经验,但毕竟也是IT科班出身了,准备用一年的事件摸索一下内容站和SEO,看看能否积累点经验,赚点小钱. 推酷-靠爬虫起家的内容站 做内容站,站长们都 ...

  6. 三、糖醋鲤鱼(Sweet and sour carp)

    糖醋鲤鱼是用鲤鱼制作的一道山东济南传统名菜,为鲁菜的代表菜品之一 ,色泽金黄,外焦内嫩,酸甜可口,香鲜味美. 菜品历史 <诗经>载:岂食其鱼,必河之鲤.<济南府志>上早有&qu ...

  7. css常用命名

    常用的CSS命名 头:header 内容:content/container 尾:footer 导航:nav 侧栏:sidebar 栏目:column 页面外围控制整体佈局宽度:wrapper 左右中 ...

  8. LeetCode编程训练 - 拓扑排序(Topological Sort)

    拓扑排序基础 拓扑排序用于解决有向无环图(DAG,Directed Acyclic Graph)按依赖关系排线性序列问题,直白地说解决这样的问题:有一组数据,其中一些数据依赖其他,问能否按依赖关系排序 ...

  9. 算法与数据结构(二) 栈与队列的线性和链式表示(Swift版)

    数据结构中的栈与队列还是经常使用的,栈与队列其实就是线性表的一种应用.因为线性队列分为顺序存储和链式存储,所以栈可以分为链栈和顺序栈,队列也可分为顺序队列和链队列.本篇博客其实就是<数据结构之线 ...

  10. 你必须知道的10个Python第三库

    1. BeautifulSoup Beautiful Soup是一个可以从HTML,XML进行提取文件的Python库,日常我们使用爬虫进行数据抓取回来之后,往往需要进行数据解析. 使用它能让你开心愉 ...