POJ1050 dp入门

先来看最大子序列和。有一串数,有正有负,如2,-1,5,4,-9,7,0,3,-5。求:这一串数中,和最大的一段。比如说,从第一个数2开始,发现下一个为-1,加下-1后和显然会变小。再往后看,第三个数是5,所以上一个-1还是要选的,这样才能加上5。哎,不看了,这样求最大和还不得累死。嘿嘿,这时DP就派上用场了。

 

设这串数为X1 X2 X3 … Xn, 用dp(i,j)表示从Xi…Xj的最大子序列和。

按照DP的思路,想办法减小问题的规模。有n个数,怎样能减少到n-1数?想办法把最后一个数Xn去掉,问题规模就能减少到n-1。

通过观察可以发现:X1…Xn的最大子序列可以分为两类:以Xn结尾、不以Xn结尾。不以Xn结尾的最大子序列,其实就是X1…Xn-1的最大子序列,发现这点很重要。

这样就有:dp( i, j ) = Max( dp( i, j-1 ), Last( j ) ).其中Last( j )表示以Xj结尾的最大子序列的和。

功夫不负有心人,终于把问题规模减少了。但是,一波未平一波又起,新的问题又出现了。Last( j )如何求?即,求以Xj结尾的最大子序列的和。再用DP求解。

Last( j )和Last( j-1 )之间的关系比较简单。Last(j )的值里面必然会包括Xj的值,到底有没有Last( j-1 )也很简单,主要取决于Last( j-1 )是正还是负。

这样就有:Last( j ) = Max( Xj,  Last( j-1) + Xj );

 状态转换方程:

dp( i, j ) = Max( dp( i, j-1 ), Last( j ) )其中:dp(i,j)表示从Xi…Xj的最大子序列和

Last( j ) = Max( Xj, Last( j-1 ) + Xj ); 其中:Last( j )表示以Xj结尾的最大子序列的和

 

       现在,回到POJ1050。想想能不能利用上面的结果?求最大子矩阵,那么只要确定了子矩阵有几行、几列即可。这样,可以枚举子矩阵的行数和列数。

比如,当子矩阵只要一行时,那么只关心它的列从哪开始到那结束就行。哦,这其实就是一个最大子序列和的问题。这一行就是这一串数,求和最大的一段。那么当子矩阵有两行时,怎么办?如何把两行变为一行?一个聪明的想法就是:把这两行按照对应的列加起来。

好了问题已经漂亮的解决了:在原矩阵中任意画出一部分,然后按照对应的列加起来,问题就转变为一个最大子序列和的问题


以上分析转载自 http://blog.csdn.net/sj13051180/article/details/6667577


#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
using namespace std;
#define NUM 105
#define Best( x , y )  ( ( (x) < (y) ) ? (y) : (x) )
int Arr[NUM][NUM];
int N;

int dp[NUM];
int Last[NUM];

int data[NUM];

int getMaxSum(int* d) //对一维数组求最大子串和 杭电1003
{
	memset(dp,0,sizeof(dp));
	memset(Last,0,sizeof(Last));
	dp[0]=d[0];
	Last[0]=d[0];
	int i;
	for(i=1;i<N;i++)
	{
		Last[i]=Best(Last[i-1]+d[i],d[i]);
		dp[i]=Best(dp[i-1],Last[i]);
	}
	return dp[N-1];
}
void Solve()
{
	int maxsum=-100000;
	memset(data,0,sizeof(data));
	int i,j,h,k;
	for(i=0;i<N;i++)
	{
		for(j=i;j<N;j++)
		{
			//compute from line i to line j
			memset(data,0,sizeof(data));
			for(h=0;h<N;h++)
			{
				for(k=i;k<=j;k++)
					data[h]+=Arr[k][h];
			}
			int tmpsum=getMaxSum(data);//求最大子串和
			maxsum=tmpsum>maxsum ? tmpsum : maxsum;
		}
	}
	printf("%d\n",maxsum);
}
int main()
{
	//freopen("poj1050.txt","r",stdin);
	memset(Arr,0,sizeof(Arr));
	while(scanf("%d",&N)!=EOF)
	{
		int i,j;
		for(i=0;i<N;i++)
			for(j=0;j<N;j++)
				scanf("%d",&Arr[i][j]);
		Solve();
	}
	return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值