poj-2479 Maximum sum

题目传送门:http://poj.org/problem?id=2479

此题是比较经典的动规题,left[]是从左到右的最大和,right[]是从右到左的最大和。最后左右相加判断最大值即可。

详细解释:

left[i]存的是从0到i的最大和,如果left[i]的值为负,那么left[i + 1]的值为a[i + 1],否则在此基础上加上left[i],因为如果left[i]为负的话,left[i + 1]加上一个负数就会减小最优值。right相同。

下面是代码:

#include <iostream>
#include <cstdio>
#include <algorithm>
using namespace std;

int a[50001], left_[50001], right_[50001];

int main()
{
	ios::sync_with_stdio(false);
	cin.tie(0);
	int t;
	cin >> t;
	while (t--)
	{
		int n;
		scanf("%d", &n);
		for (int i = 0; i < n; i++)
			scanf("%d", &a[i]);
		left_[0] = a[0];
		for (int i = 1; i < n; i++)
		{
			if (left_[i - 1] < 0)
				left_[i] = a[i];
			else
				left_[i] = left_[i - 1] + a[i];
		}
		for (int i = 1; i < n; i++)
			left_[i] = max(left_[i], left_[i - 1]);
		right_[n - 1] = a[n - 1];
		for (int j = n - 2; j >= 0; j--)
		{
			if (right_[j + 1] < 0)
				right_[j] = a[j];
			else
				right_[j] = right_[j + 1] + a[j];
		}
		for (int i = n - 2; i >= 0; i--)
			right_[i] = max(right_[i + 1], right_[i]);
		int res = -100000000;
		for (int i = 1; i < n; i++)
			res = max(res, left_[i - 1] + right_[i]);
		cout << res << endl;
	}
	return 0;
}


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值