codeforces 407D Largest Submatrix 3

题意

找出最大子矩阵,须满足矩阵内的元素互不相等。

题解

官方做法

http://codeforces.com/blog/entry/11333

  • \(O(n^6)\)

    • 枚举子矩阵,暴力check。
  • \(O(n^4)\)
    • 枚举上下边界,双指针。
  • \(O(n^3log_2n)\)
    • 假设当前上边界 \(up\), 下边界 \(down\),\(R_i\) 表示当 \(i\) 为左边界时,右边界最大是 \(R_i\)。
    • 当 \(down->down+1\) 时,\(R_i\) 要么不变,要么减小。
    • 减小的情况:\(a_{down+1, j}(i<=j<=R_i)\) 在矩阵 \((up, i, down, R_i)\) 中出现。
    • 因此对于每个 \(a_{down+1, j}(i<=j<=R_i)\),我们需要寻找在矩阵 \((up, i, down, R_i)\) 中出现的,从左边最靠近 \(j\),和从右边最靠近 \(j\) 的两个位置。这个可以用set维护。
  • \(O(n^3)\)
    • 如何优化掉 \(O(n^3log_2n)\) 做法的 \(log_2n\) ?
    • 从大到小枚举 \(up\)。当 \(up->up-1\) 时,\(O(n^2)\) 更新 \((up, 1, n, m)\) 中所有点的从左边最靠近,和从右边最靠近的两个位置。

区间dp做法

\(f_{i, l, r}\)表示下边界是 \(i\), 左右边界是 \(l, r\) 时,上边界的值。

\(f_{i, l, r}=max\{f_{i-1, l, r}, f_{i, l+1, r}, f_{i, l, r-1}, a_{i, l}与a_{1-i, r}的限制, a_{i, r}与a_{1-i, l}的限制\}\)

可以这样理解:\(f_{i, l, r}\) 需要维护A、B、C、D、E、F这六个块之间的关系,\(f_{i-1, l, r}\) 只维护了A、B、C之间的关系,\(f_{i, l+1, r}\) 只维护了B、C、E、F之间的关系,\(f_{i, l, r-1}\) 只维护了A、B、D、E之间的关系,剩下的还需维护A与F、C与D、D与F的关系。

// 代码实现可以省掉第一维。

代码

\(O(n^3log_2n)\)做法

T了 不知道有没有写错

#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
#define rep(i, a, b) for(int i=(a); i<(b); i++)
#define sz(x) (int)x.size()
#define de(x) cout<< #x<<" = "<<x<<endl
#define dd(x) cout<< #x<<" = "<<x<<" "
typedef long long ll;
typedef pair<int, int> pii;
typedef vector<int> vi; const int N=404;
int n,m;
int a[N][N], R[N], cnt[N*N];
set<int> pos[N*N]; int solve() {
int ans=1;
rep(up,1,n+1) {
rep(i,0,N*N) pos[i].clear();
rep(i,0,m+1) R[i]=m;
rep(down,up,n+1) {
int r=0;
rep(i,1,m+1) {
while(r<m&&!cnt[a[down][r+1]]) {
++r;
++cnt[a[down][r]];
}
R[i]=min(R[i], r);
--cnt[a[down][i]];
}
rep(j,1,m+1) {
int c=a[down][j];
auto it1=pos[c].upper_bound(j);
auto it2=pos[c].lower_bound(j);
if(it1!=pos[c].begin()) {
--it1;
if(*it1<=j) R[*it1]=min(R[*it1], j-1);
}
if(it2!=pos[c].end()) {
if(j<=*it2) R[j]=min(R[j], *it2-1);
}
}
for(int i=m-1;i;--i) R[i]=min(R[i], R[i+1]);
rep(i,1,m+1) if(i<=R[i]) ans=max(ans, (down-up+1)*(R[i]-i+1));
rep(i,1,m+1) pos[a[down][i]].insert(i);
}
}
return ans;
} int main() {
while(~scanf("%d%d",&n,&m)) {
///read
rep(i,1,n+1) rep(j,1,m+1) scanf("%d",&a[i][j]);
///solve
printf("%d\n",solve());
}
return 0;
}

\(O(n^3)\) 做法

#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
#define rep(i, a, b) for(int i=(a); i<(b); i++)
#define sz(x) (int)x.size()
#define de(x) cout<< #x<<" = "<<x<<endl
#define dd(x) cout<< #x<<" = "<<x<<" "
typedef long long ll;
typedef pair<int, int> pii;
typedef vector<int> vi; const int N=404;
int n, m;
int a[N][N], p[N*N], l[N][N], r[N][N], R[N]; void Min(int &a, int b) {
if(a>b) a=b;
}
void Max(int &a, int b) {
if(a<b) a=b;
} int solve() {
int ans=1;
rep(i,0,n+1) rep(j,0,m+1) l[i][j]=0, r[i][j]=m+1;
for(int u=n;u;--u) {
rep(i,1,m+1) p[a[u][i]]=0;
rep(i,1,m+1) l[u][i]=p[a[u][i]], p[a[u][i]]=i;
rep(i,1,m+1) p[a[u][i]]=m+1;
for(int i=m;i;--i) r[u][i]=p[a[u][i]], p[a[u][i]]=i;
memset(p,0,sizeof(p));
rep(i,1,m+1) {
p[a[u][i]]=i;
rep(v,u+1,n+1) Max(l[v][i], p[a[v][i]]);
}
rep(i,0,N*N) p[i]=m+1;
for(int i=m;i;--i) {
p[a[u][i]]=i;
rep(v,u+1,n+1) Min(r[v][i], p[a[v][i]]);
}
rep(i,1,m+1) R[i]=m+1;
rep(v,u,n+1) {
rep(i,1,m+1) {
Min(R[l[v][i]], i);
Min(R[i], r[v][i]);
}
for(int i=m-1;i;--i) Min(R[i], R[i+1]);
rep(i,1,m+1) {
Max(ans, (v-u+1)*(R[i]-i));
}
}
}
return ans;
} int main() {
while(~scanf("%d%d",&n,&m)) {
///read
rep(i,1,n+1) rep(j,1,m+1) scanf("%d",&a[i][j]);
///solve
printf("%d\n",solve());
}
return 0;
}

区间dp做法

#include<bits/stdc++.h>
using namespace std;
#define fi first
#define se second
#define mp make_pair
#define pb push_back
#define rep(i, a, b) for(int i=(a); i<(b); i++)
#define sz(x) (int)x.size()
#define de(x) cout<< #x<<" = "<<x<<endl
#define dd(x) cout<< #x<<" = "<<x<<" "
typedef long long ll;
typedef pair<int, int> pii;
typedef vector<int> vi; const int N=404;
int n,m;
int a[N][N], f[N][N], p[N][160004]; int main() {
while(~scanf("%d%d",&n,&m)) {
rep(i,1,n+1) rep(j,1,m+1) scanf("%d",&a[i][j]);
memset(f,0,sizeof(f));
memset(p,0,sizeof(p));
int ans=1;
rep(i,1,n+1) {
rep(len,1,m+1) {
for(int l=1, r=len;r<=m;++l, ++r) {
if(l==r) {
f[l][r]=max(f[l][r], p[l][a[i][l]]);
} else {
f[l][r]=max(f[l][r], max(f[l][r-1], f[l+1][r]));
f[l][r]=max(f[l][r], max(p[l][a[i][r]], p[r][a[i][l]]));
if(a[i][l]==a[i][r]) f[l][r]=i;
}
ans=max(ans, (i-f[l][r])*(r-l+1));
}
}
rep(j,1,m+1) p[j][a[i][j]]=i;
}
printf("%d\n",ans);
}
return 0;
}

codeforces 407D Largest Submatrix 3的更多相关文章

  1. Largest Submatrix(动态规划)

    Largest Submatrix Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others ...

  2. POJ-3494 Largest Submatrix of All 1’s (单调栈)

    Largest Submatrix of All 1’s Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 8551   Ac ...

  3. hdu 2870 Largest Submatrix(平面直方图的最大面积 变形)

    Problem Description Now here is a matrix with letter 'a','b','c','w','x','y','z' and you can change ...

  4. Largest Submatrix of All 1’s

    Given a m-by-n (0,1)-matrix, of all its submatrices of all 1’s which is the largest? By largest we m ...

  5. Largest Submatrix of All 1’s(思维+单调栈)

    Given a m-by-n (0,1)-matrix, of all its submatrices of all 1's which is the largest? By largest we m ...

  6. POJ 3494 Largest Submatrix of All 1’s 单调队列||单调栈

    POJ 3494 Largest Submatrix of All 1’s Description Given a m-by-n (0,1)-matrix, of all its submatrice ...

  7. POJ - 3494 Largest Submatrix of All 1’s 单调栈求最大子矩阵

    Largest Submatrix of All 1’s Given a m-by-n (0,1)-matrix, of all its submatrices of all 1’s which is ...

  8. HDU 2870 Largest Submatrix (单调栈)

    http://acm.hdu.edu.cn/showproblem.php? pid=2870 Largest Submatrix Time Limit: 2000/1000 MS (Java/Oth ...

  9. MINSUB - Largest Submatrix

    MINSUB - Largest Submatrix no tags  You are given an matrix M (consisting of nonnegative integers) a ...

随机推荐

  1. 我与ARM的那些事儿1初识ARM

    最近一直在研究ARM,说到ARM,我们首先想到了是三星.高通等公司,这些公司都制造CPU的,其实ARM也是一家公司,只不过它是提供最核心的逻辑电路,而且它的赚钱方式是与其他公司进行双赢的!你卖出多少芯 ...

  2. Mac下PHP+Apache+MySQL环境搭建

    一.启动Apache 有两种方法 1.打开网络共享 打开"系统偏好设置"->"共享",在"互联网共享"那一项前面打√. 2.打开终端, ...

  3. 八: 操作提示(wxml 即将废弃)

    首先需要注意的是 wxml的这些属性将要被废弃,不过可以看两眼.不愿意看的可以看下一章节同样是操作回馈只不过是js版的哦.   一.action-sheet 操作菜单 从屏幕底下出来菜单. 这里不用w ...

  4. SpringBoot 之Quartz的使用

    对于Quartz的使用,还是想说一句,SpringBoot真的很好用啊! 第一步:当然是引入依赖啦 <parent> <groupId>org.springframework. ...

  5. java.lang.SecurityException: Permission Denial: reading com.android.providers.media.MediaProvider uri content://media/external/images/media/20 from pid=711, uid=10074 requires android.permission.READ_

    java.lang.SecurityException: Permission Denial: reading com.android.providers.media.MediaProvider ur ...

  6. groovy类、构造函数、方法

    数据类型:groovy支持Java语言规范定义的数据类型 类:与Java类的主要区别 1.没有可见修饰符的类或者方法是自动公开的 2.类不需要与源文件定义相同名称,但是默认规定定义一样 3.一个源文件 ...

  7. CodeForces 598A(水)

    还是要注意int和long long的范围,以及double型的问题 pow函数经常会报一个double型的错,参考这篇文章 http://blog.csdn.net/lawrencesgj/arti ...

  8. C#学习笔记(基础知识回顾)之值类型和引用类型

    一:C#把数据类型分为值类型和引用类型 1.1:从概念上来看,其区别是值类型直接存储值,而引用类型存储对值的引用. 1.2:这两种类型在内存的不同地方,值类型存储在堆栈中,而引用类型存储在托管对上.存 ...

  9. 浏览器根对象window之history

    1. history(H5) Window.history保存用户在一个会话期间的网站访问记录,用户每次访问一个新的URL即创建一个新的历史记录. 1.1 length 返回浏览器历史列表中的 URL ...

  10. Spring Tech

    1.Spring中AOP的应用场景.Aop原理.好处? 答:AOP--Aspect Oriented Programming面向切面编程:用来封装横切关注点,具体可以在下面的场景中使用: Authen ...