【♂Not The Triumph♂O(∩_∩)O哈哈~But The Struggle♂】

竞赛决不是捷径,它只是另一种艰辛的生活方式。得到与失去,只有时间会去评判;成功与失败,只有历史能去仲裁。我不会永远成功,正如我不会永远失败一样

  C++博客 :: 首页 :: 联系 ::  :: 管理
  6 Posts :: 239 Stories :: 25 Comments :: 0 Trackbacks

常用链接

留言簿(7)

我参与的团队

搜索

  •  

积分与排名

  • 积分 - 108444
  • 排名 - 229

最新评论

阅读排行榜

评论排行榜

Description
【题目说明】
   此题中出现的所有数全为整数
【题目背景】
   SubRaY有一天得到一块西瓜,是长方体形的....
【题目描述】
   SubRaY发现这块西瓜长m厘米,宽n厘米,高h厘米.他发现如果把这块西瓜平均地分成m*n*h块1立方厘米的小正方体,那么每一小块都会有一个营养值(可能为负,因为西瓜是有可能坏掉的,但是绝对值不超过200).
现在SubRaY决定从这m*n*h立方厘米的西瓜中切出mm*nn*hh立方厘米的一块小西瓜(一定是立方体形,长宽高均为整数),然后吃掉它.他想知道他最多能获得多少营养值.(0<=mm<=m,0<=nn<=n,0<=hh<=h.mm,nn,hh 的值由您来决定).
换句话说,我们希望从一个m*n*h的三维矩阵中,找出一个三维子矩阵,这个子矩阵的权和最大。


Input
首行三个数h,m,n(注意顺序),分别表示西瓜的高,长,宽.以下h部分,每部分是一个m*n的矩阵,第i部分第j行的第k个数表示西瓜第i层,第j行第k列的那块1立方厘米的小正方体的营养值。

Output
SubRaY所能得到的最大营养值

Sample Input
2 3 4
4 1 2 8
0 5 -48 4
3 0 1 9
2 1 4 9
1 0 1 7
3 1 2 8

Sample Output
45

Hint
【数据范围】
对于30%的数据,h=1,1<=m,n<=10
对于全部的数据,1<=h<=32,1<=m,n<=50,保证h<=m,n。


分析:

  把三维的每个高(竖的下来的那些列,不是y坐标那个列)的和存放到二维数组中,再用二维的最大子矩阵即可。(好像废话^_^)


【参考程序】:
#include<cstring>
#include
<cstdio>
#include
<iostream>
#define INF -25000000
using namespace std;
int sum[60][60][60],he[60][60][60],s[60][60];
int h,m,n,ans;
int Maxsum1(int *a)
{
    
int sums=INF,ss=0;
    
for (int i=1;i<=n;i++)
    {
        
if (ss>0) ss+=a[i];
        
else ss=a[i];
        
if (ss>sums) sums=ss;
    }
    
return sums;
}
int Maxsum2(int a[][60])
{
    
int sums=INF,ss,F[60];
    
for (int i=1;i<=m;i++)
    {
        memset(F,
0,sizeof(F));
        
for (int j=i;j<=m;j++)
        {
            
for (int k=1;k<=n;k++) F[k]+=a[j][k];
            
/*ss=0;
            for (int k=1;k<=n;k++)
                if (ss>0) ss+=F[k];
                else ss=F[k];
*/
            ss
=Maxsum1(F);
            
if (ss>sums) sums=ss;
        }
    }
    
return sums;
}
int Maxsum3()
{
    
int sums=INF,ss;
    
for (int i=1;i<=h;i++)
        
for (int j=i;j<=h;j++)
        {
            
for (int i1=1;i1<=m;i1++)
                
for (int j1=1;j1<=n;j1++)
                    s[i1][j1]
=sum[j][i1][j1]-sum[i-1][i1][j1];
            ss
=Maxsum2(s);
            
if (sums<ss) sums=ss;
        }
    
return sums;
}
int main()
{
    scanf(
"%d%d%d",&h,&m,&n);
    
for (int i=1;i<=h;i++)
        
for (int j=1;j<=m;j++)
            
for (int k=1;k<=n;k++)
                scanf(
"%d",&he[i][j][k]);
    
for (int i=1;i<=m;i++)
        
for (int j=1;j<=n;j++)
            sum[
1][i][j]=he[1][i][j];
    
for (int i=2;i<=h;i++)
        
for (int j=1;j<=m;j++)
            
for (int k=1;k<=n;k++)
                sum[i][j][k]
=sum[i-1][j][k]+he[i][j][k];
    ans
=Maxsum3();
    printf(
"%d\n",ans);
    
return 0;
}
posted on 2009-08-20 11:33 开拓者 阅读(577) 评论(0)  编辑 收藏 引用 所属分类: 动态规划&背包经典习题

只有注册用户登录后才能发表评论。
网站导航: 博客园   IT新闻   BlogJava   知识库   博问   管理