hihocoder1580 Matrix
题目链接:(vjudge)戳我
从今天开始不咕咕地填坑啦
考虑一般的求最大子矩阵和。。。我们一般都是DP,或者直接上悬线法递推。
下面附一个DP的代码:
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#define MAXN 310
using namespace std;
int n,m,p,all,ans;
int a[MAXN][MAXN],dp[MAXN],minn[MAXN],sum[MAXN];
int main()
{
#ifndef ONLINE_JUDGE
freopen("ce.in","r",stdin);
#endif
while(scanf("%d%d%d",&n,&m,&p)!=EOF)
{
memset(a,0,sizeof(a));
memset(dp,0,sizeof(dp));
memset(minn,0x3f,sizeof(minn));
all=0,ans=-2147483647;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%d",&a[i][j]),all+=a[i][j];
for(int l=1;l<=n;l++)
{
memset(sum,0,sizeof(sum));
for(int r=l;r<=n;r++)
{
for(int i=1;i<=m;i++)
{
sum[i]+=a[r][i];
if(l==r) minn[i]=a[r][i];
else minn[i]=min(minn[i],a[r][i]);
if(i==1) dp[i]=sum[i];
else dp[i]=max(sum[i],dp[i-1]+sum[i]);
ans=max(ans,dp[i]);
}
}
}
printf("%d\n",ans);
}
return 0;
}
然后这个题就是最大子矩阵和的变形。要求必须修改一个。
我们新开一个数组dp2[i]来记录第一维在\([l,r]\)范围内(这个做右端点是需要\(n^2\)枚举的),第二维计算到i的时候,修改了一个位置为p的最大子矩阵和。
因为要修改之后尽量大,所以我们转移的时候需要维护当前\(i\)这一列最小的矩形中最小的元素,那么带修改的自然就是减去这个最小值然后加上p了。
转移最大子矩形和的时候注意分类讨论,分一类为继承前一列,一类为自己新开一个。带修改的还需要分类讨论如果继承前一列,那个带修改的位置到底是在自己这一列还是在前面列中。
然后还有一点就是因为我们修改的这个位置不一定被选入到我们的最大子矩阵和的矩阵里,所以每个不修改的状态也要记录一遍。(当然要特判一下,所有数取满是非法情况要排除掉啦!)
然后每种情况都和ans取一个最大值。
具体看代码吧。代码如下:
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#define MAXN 310
using namespace std;
int n,m,p,all,ans;
int a[MAXN][MAXN],dp1[MAXN],dp2[MAXN],minn[MAXN],sum[MAXN];
int main()
{
#ifndef ONLINE_JUDGE
freopen("ce.in","r",stdin);
#endif
while(scanf("%d%d%d",&n,&m,&p)!=EOF)
{
memset(a,0,sizeof(a));
memset(dp1,0,sizeof(dp1));
memset(dp2,0,sizeof(dp2));
memset(minn,0x3f,sizeof(minn));
all=0,ans=-2147483647;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
scanf("%d",&a[i][j]),all+=a[i][j];
for(int l=1;l<=n;l++)
{
memset(sum,0,sizeof(sum));
for(int r=l;r<=n;r++)
{
for(int i=1;i<=m;i++)
{
sum[i]+=a[r][i];
if(l==r) minn[i]=a[r][i];
else minn[i]=min(minn[i],a[r][i]);
if(i==1) dp1[i]=sum[i];
else dp1[i]=max(sum[i],dp1[i-1]+sum[i]);
if(i==1) dp2[i]=sum[i]-minn[i]+p;
else dp2[i]=max(dp2[i-1]+sum[i],max(dp1[i-1]+sum[i]-minn[i]+p,sum[i]-minn[i]+p));
if(dp1[i]!=all)
ans=max(ans,dp1[i]);
ans=max(ans,dp2[i]);
printf("l=%d r=%d i=%d ans=%d\n",l,r,i,ans);
}
}
}
printf("%d\n",ans);
}
return 0;
}
hihocoder1580 Matrix的更多相关文章
- angular2系列教程(十一)路由嵌套、路由生命周期、matrix URL notation
今天我们要讲的是ng2的路由的第二部分,包括路由嵌套.路由生命周期等知识点. 例子 例子仍然是上节课的例子:
- Pramp mock interview (4th practice): Matrix Spiral Print
March 16, 2016 Problem statement:Given a 2D array (matrix) named M, print all items of M in a spiral ...
- Atitit Data Matrix dm码的原理与特点
Atitit Data Matrix dm码的原理与特点 Datamatrix原名Datacode,由美国国际资料公司(International Data Matrix, 简称ID Matrix)于 ...
- Android笔记——Matrix
转自:http://www.cnblogs.com/qiengo/archive/2012/06/30/2570874.html#translate Matrix的数学原理 在Android中,如果你 ...
- 通过Matrix进行二维图形仿射变换
Affine Transformation是一种二维坐标到二维坐标之间的线性变换,保持二维图形的"平直性"和"平行性".仿射变换可以通过一系列的原子变换的复合来 ...
- [LeetCode] Kth Smallest Element in a Sorted Matrix 有序矩阵中第K小的元素
Given a n x n matrix where each of the rows and columns are sorted in ascending order, find the kth ...
- [LeetCode] Longest Increasing Path in a Matrix 矩阵中的最长递增路径
Given an integer matrix, find the length of the longest increasing path. From each cell, you can eit ...
- [LeetCode] Search a 2D Matrix II 搜索一个二维矩阵之二
Write an efficient algorithm that searches for a value in an m x n matrix. This matrix has the follo ...
- [LeetCode] Search a 2D Matrix 搜索一个二维矩阵
Write an efficient algorithm that searches for a value in an m x n matrix. This matrix has the follo ...
随机推荐
- VMWare中三种网络连接模式的区别
VMWare中有桥接.NAT.host-only三种网络连接模式,在搭建伪分布式集群时,需要对集群的网络连接进行配置,而这一操作的前提是理解这三种网络模式的区别. 参考以下两篇文章可以更好的理解: V ...
- [JS] 让人犯晕的JavaScript变量赋值
变量赋值 本文转载自http://hellobug.github.io/blog/javascript-variable-assignment/ 开始之前先来几个例子,确保起始点是晕的状态- :P 例 ...
- Linux实战教学笔记23:Inotify事件监控工具
第二十三节 inotify事件监控工具 标签(空格分隔): Linux实战教学笔记-陈思齐 ---本教学笔记是本人学习和工作生涯中的摘记整理而成,此为初稿(尚有诸多不完善之处),为原创作品,允许转载, ...
- 阻塞和唤醒线程——LockSupport功能简介及原理浅析
目录 1.LockSupport功能简介 1.1 使用wait,notify阻塞唤醒线程 1.2 使用LockSupport阻塞唤醒线程 2. LockSupport的其他特色 2.1 可以先唤醒线程 ...
- 第十一条理解objc_masgSend的作用
Objetive-C最基本的的东西就是它的消息机制.Objective-C运行时的最基本的东西就是 objc_msgSend, 它就是负责发送一个消息给对象的C函数. 当你写下面这样的代码时: ...
- JAVA本地调用(JNI- java调用c)
记录一下工作内容,对术语了解不多,暂且这样记着吧. java调用c 一.写jni的步骤如下: 1.创建java类,定义接口函数,使用native修饰: 2.将java类编译成class: 3.将cl ...
- js 数字游戏
在某网站看到一道js题,觉得有点意思 Some numbers have funny properties. For example: 89 --> 8¹ + 9² = 89 * 1 695 - ...
- Android中的Handler介绍
一.Handler的定义: 主要接受子线程发送的数据, 并用此数据配合主线程更新UI. 解释: 当应用程序启动时,Android首先会开启一个主线程 (也就是UI线程) , 主线程为管理界面中的UI控 ...
- PHP Windows系统下调用OpenOffice
项目需要把用户上传的word文档转换为pdf文件,方便用户浏览.经过谷歌百度找到PHP可以使用COM组件调用微软的openoffice来实现文档转换 1,安装OpenOffice 安装OpenOffi ...
- 免秘钥oracel官方下载jdk
wget --no-check-certificate --no-cookies --header "Cookie: oraclelicense=accept-securebackup-co ...