洛谷 P1736 创意吃鱼法 Label:dp || 前缀和
题目描述
回到家中的猫猫把三桶鱼全部转移到了她那长方形大池子中,然后开始思考:到底要以何种方法吃鱼呢(猫猫就是这么可爱,吃鱼也要想好吃法 ^_*)。她发现,把大池子视为01矩阵(0表示对应位置无鱼,1表示对应位置有鱼)有助于决定吃鱼策略。
在代表池子的01矩阵中,有很多的正方形子矩阵,如果某个正方形子矩阵的某条对角线上都有鱼,且此正方形子矩阵的其他地方无鱼,猫猫就可以从这个正方形子矩阵“对角线的一端”下口,只一吸,就能把对角线上的那一队鲜鱼吸入口中。
猫猫是个贪婪的家伙,所以她想一口吃掉尽量多的鱼。请你帮猫猫计算一下,她一口下去,最多可以吃掉多少条鱼?
输入输出格式
输入格式:
有多组输入数据,每组数据:
第一行有两个整数n和m(n,m≥1),描述池塘规模。接下来的n行,每行有m个数字(非“0”即“1”)。每两个数字之间用空格隔开。
对于30%的数据,有n,m≤100
对于60%的数据,有n,m≤1000
对于100%的数据,有n,m≤2500
输出格式:
只有一个整数——猫猫一口下去可以吃掉的鱼的数量,占一行,行末有回车。
输入输出样例
4 6
0 1 0 1 0 0
0 0 1 0 1 0
1 1 0 0 0 1
0 1 1 0 1 0
3
说明
右上角的
1 0 0 0 1 0 0 0 1
解法一
这道题虽然是动态规划,但也有非动态规划的做法,看到题解里没有这种做法,就给大家发一下(虽然慢了一点,但比较容易理解):用前缀和、差分,首先对矩阵求前缀和,之后从(0,0)向(n,m)扫描,每扫描到一个点,就向这个点的左上方(i-k,j-k,1<=k<=n)
扫描,如果扫描到0或从(i-k,j-k)到(i,j)这个矩阵的差分(即其中鱼的个数)大于k+1(矩阵对角线长度)则终止扫描,此时k就是从点(i,j)向左上方能吃到的鱼的数量,扫描完之后将整个矩阵左右翻转再进行一次扫描,两次扫描扫到的最大值就是结果。
代码:
#include<iostream>
#include<cstdio>
#include<cstring>
#include<algorithm>
#define min3(a,b,c) min(a,min(b,c))
using namespace std;
int ans,a[][],b[][],sum[][],N,M,x,y;
void zheng(){
memset(sum,,sizeof(sum));
for(int i=;i<=N;i++){
for(int j=;j<=M;j++)
sum[i][j]=sum[i][j-]+a[i][j];
for(int j=;j<=M;j++)
sum[i][j]+=sum[i-][j];
} for(int i=;i<=N;i++)
for(int j=;j<=M;j++)
if(a[i][j]==){
int k=;
while((k<min(i,j))&&(a[i-k][j-k]==)
&&(sum[i][j]-sum[i-k-][j]-sum[i][j-k-]+sum[i-k-][j-k-])==(k+)){
// printf("i=%d j=%d k=%d\n",i,j,k);
k++;
} ans=max(ans,k);
} }
void change(){
for(int i=;i<=N;i++)
for(int j=;j<=(M+)/;j++)
swap(a[i][j],a[i][M-j+]);
return;
}
int main(){
// freopen("01.txt","r",stdin);
scanf("%d%d",&N,&M);
for(int i=;i<=N;i++)
for(int j=;j<=M;j++)
scanf("%d",&a[i][j]); zheng();
change();
/* for(int i=1;i<=N;i++){
for(int j=1;j<=M;j++)
printf("%-3d",sum[i][j]);
puts("");
}*/ zheng(); printf("%d\n",ans);
return ;
}
解法二
动态规划1,f[i][j]表示以(i,j)为顶点的最大子矩阵中能吃到的鱼的条数。
f[i][j]可以从f[i-1][j-1]或f[i-1][j+1]转移而来,取其中最大值。
每次转移需要横向和纵向检查0的个数,取两者最小值k即得到了最大子矩阵中鱼的个数。
C代码:
#include<stdio.h>
int n,m,i,j,k,k1,k2,max,temp,f[][];
int main(void)
{
while(scanf("%d%d",&n,&m)==)
{
max=;//答案
for(i=;i<=n;i++)
{
for(j=;j<=m;j++)scanf("%d",&f[i][j]);//先输入一行,因为下面要横向检查0的个数。
for(j=;j<=m;j++)
{
if(f[i][j]==)
{
temp=f[i-][j-];//检查左上方
if(temp>= && temp+>f[i][j])
{
for(k1=;k1<=temp;k1++)//横向往左检查0的个数k1,最多为temp个
if(f[i][j-k1]>=) break;
for(k2=;k2<=temp;k2++)//纵向往上检查k2……
if(f[i-k2][j]>=) break;
f[i][j]=k1<k2?k1:k2; //取k1、k2较小的那个,就是满足条件的最大子矩阵。
}
temp=f[i-][j+];//检查右上方
if(temp>= && temp+>f[i][j])
{
for(k1=;k1<=temp;k1++)
if(f[i][j+k1]>=) break;//唯一不同是这里的j+k1而不是j-k1,因为是往右检查
for(k2=;k2<=temp;k2++)
if(f[i-k2][j]>=) break;
f[i][j]=k1<k2?k1:k2;
}
}
if(f[i][j]>max) max=f[i][j];
}
}
printf("%d\n",max);
}
return ;
}
动态规划2:
暴力dp。
首先分类,fl[i,j]表示由左上开始的对角线到(i,j)为终点的最大长度,fr[i,j]为从右上开始的对角线到(i,j)为终点的最大长度。
可以观察到fl[i,j]可以从f[i-1,j-1]转移而来,fr[i,j]可由fr[i-1,j+1]转移而来,当转移完毕后,搜索以当前对角线长为边长正方形上该点的临边有没有鱼,如果有鱼,那么鱼到该点的距离为以该点为终点的对角线长。
#include<iostream>
#include<cstring>
#include<cstdio>
using namespace std;
int a[][],fl[][],fr[][];
int main(){
int n,m;
cin>>n>>m;
for(int i = ;i<=n;i++)
for(int j = ;j<=m;j++)scanf("%d",&a[i][j]);
memset(fl,,sizeof(fl));
memset(fr,,sizeof(fr));
for(int i = ;i<=n;i++)//搜索fl
for(int j = ;j<=m;j++)
if(a[i][j]){
fl[i][j] = fl[i-][j-]+;
for(int k = ;k<fl[i][j];k++)if(a[i-k][j]||a[i][j-k]){//验证当前解是否可行
fl[i][j] = k;
break;
}
}
for(int i = ;i<=n;i++)//搜索fr
for(int j = m;j;j--)
if(a[i][j]){
fr[i][j] = fr[i-][j+]+;
for(int k = ;k<fr[i][j];k++)if(a[i-k][j]||a[i][j+k]){//验证当前解是否可行
fr[i][j] = k;
break;
}
}
int ans = ;
for(int i = ;i<=n;i++)
for(int j = ;j<=m;j++)ans = max(ans,max(fl[i][j],fr[i][j]));
cout<<ans<<endl;
return ;
}
同类题目http://www.cnblogs.com/radiumlrb/p/5808285.html
洛谷 P1736 创意吃鱼法 Label:dp || 前缀和的更多相关文章
- 洛谷 P1736 创意吃鱼法
题目描述 题目链接:https://www.luogu.org/problemnew/show/P1736 回到家中的猫猫把三桶鱼全部转移到了她那长方形大池子中,然后开始思考:到底要以何种方法吃鱼呢( ...
- 洛谷P1736 创意吃鱼法
题目描述 回到家中的猫猫把三桶鱼全部转移到了她那长方形大池子中,然后开始思考:到底要以何种方法吃鱼呢(猫猫就是这么可爱,吃鱼也要想好吃法 ^_*).她发现,把大池子视为01矩阵(0表示对应位置无鱼,1 ...
- 洛谷P1736 创意吃鱼法 dp
正解:dp 解题报告: 早就想写dp的题目辣!我发现我的dp好差啊QAQ所以看到列表的小朋友写dp的题目就跟着他们的步伐做下题好辣QwQ 这题的话没有那——么难,大概说下趴QwQ 首先说下题意 前面一 ...
- 洛谷 P1736 创意吃鱼法(多维DP)
题目描述 回到家中的猫猫把三桶鱼全部转移到了她那长方形大池子中,然后开始思考:到底要以何种方法吃鱼呢(猫猫就是这么可爱,吃鱼也要想好吃法 ^_*).她发现,把大池子视为01矩阵(0表示对应位置无鱼,1 ...
- P1387 最大正方形 && P1736 创意吃鱼法(DP)
题目描述 在一个n*m的只包含0和1的矩阵里找出一个不包含0的最大正方形,输出边长. 输入输出格式 输入格式: 输入文件第一行为两个整数n,m(1<=n,m<=100),接下来n行,每行m ...
- Luogu P1736 创意吃鱼法【dp】By cellur925
题目传送门 题意:给出一个01矩阵,找出一条对角线,使得对角线上的元素都为1,而对角线所在矩阵其他元素均为0,使得这样的对角线最长. 状态:$f[i][j]$表示以($i$,$j$)为对角线端点的最长 ...
- P1387 最大正方形&&P1736 创意吃鱼法
P1387 最大正方形 P1736 创意吃鱼法 两道类似的$DP$ 转移方程基本上类似于$f[i][j]=min(f[i-1][j-1],min(f[i][j-1],f[i-1][j]))$ 考虑构成 ...
- P1736 创意吃鱼法 图的DP
题目描述 回到家中的猫猫把三桶鱼全部转移到了她那长方形大池子中,然后开始思考:到底要以何种方法吃鱼呢(猫猫就是这么可爱,吃鱼也要想好吃法 ^_*).她发现,把大池子视为01矩阵(0表示对应位置无鱼,1 ...
- P1736 创意吃鱼法
题目描述 回到家中的猫猫把三桶鱼全部转移到了她那长方形大池子中,然后开始思考:到底要以何种方法吃鱼呢(猫猫就是这么可爱,吃鱼也要想好吃法 ^_*).她发现,把大池子视为01矩阵(0表示对应位置无鱼,1 ...
随机推荐
- jquery ajax beforeSend 提交等待问题
需要使用异步加载async : true 否则不会出现等待效果 $.ajax({ url : $('#form').attr("action"), data: $('#form') ...
- [Unity3D]导入模型并且设置相应的属性
将资源拷贝到Assets中并设置 首先确保法线贴图的属性如下: 设置基本贴图和法线贴图 因为要发布到移动端,设置shader属性: 为了设置更好的反光效果,添加cubemap: 添加新的Materia ...
- Unicode文件读取 出现隐藏字符 (大坑)
C#读取文件..分析时发现应该15位的.. str.Lenght 却 16位.. 字符串复制出来一位位的数..就是15位.. 纳闷中突然想起来会不会是隐藏字符.. 输出 str[0].ToBytes( ...
- Property和attribute的区别[转]
Attribute和Property都可以翻译成“属性”,有的地方用Attribute表示“属性”,有的地方又在用Property,初 学者常常在这两个单词间“迷失”,甚至认为二者没有区别,是一样的. ...
- php数组函数分析--array_column
array_column 官方地址:array_column array_column 只能在 PHP版本5.5以上的运行,5.3是不支持这个函数的.如果5.3使用会报: Fatal error: C ...
- hibernate4学习
1. 安装hibernatetools插件 2. 这个是篇测试文档 来自为知笔记(Wiz)
- Shell入门教程:Shell变量
变量 是一种很“弱”的变量,默认情况下,一个变量保存一个串,Shell不关心这个串是什么含义.所以若要进行数学运算,必须使用一些命令例如 let.declare.expr.双括号等. Shell变量可 ...
- 安卓APP关于切图标
bin res drawable-hdpi drawable-ldpi drawable-mdpi drawable-nodpi drawable-xhdpi drawable-xxhdpi x越大代 ...
- 用python修改haproxy配置文件
需求: 当用户输入域名的时候,显示出来下面的记录 当用户需要输入添加纪录的时候,添加到你需要的那个域名下面 global log 127.0.0.1 local2 daemon maxconn 256 ...
- VIM下的跳转练习
在vim下可以使用常用的箭头键 但是 还有其它键可以让你更快的达到目标 hjkl 这是代替箭头键功能的 H M L 跳到屏幕的顶上 中间 下方 w 跳到下一个单词的开始e 跳到单词的结束b 向后跳 g ...