HAUTOJ 1282--ykc想吃好吃的

HAUTOJ--ykc想吃好吃的

http://218.28.220.249:50015/JudgeOnline/problem.php?id=1282


题目描述
一天,ykc在学校闲的无聊,于是决定上街买点吃的,ykc很懒,本来就不是很像逛街,于是找来了czl帮他买,这里应该有滑稽,而czl也不愿为ykc买东西吃,但是ykc很强势,非让他去买,呢没办法了,然而czl还有很多事要做,没呢么多时间帮ykc,而这条小吃街又很长,有n家店,n有50000这么大,并且这n家店的商品价值有所不同(要知道,商品的价值可能为负,哈哈,很神奇吧,但是czl肯定不会傻到赔钱,所以你懂的),哇,czl要疯了,他不想逛这么久啊,他还有个毛病,他只会连续的逛若干家店,并且由于这条街的店很多,所以肯定不会是一条直线,换句话说就是首尾相连,即第n家店和第一家店是连在一起的,然而ykc希望czl买的东西价值最大,不然就会不开心,于是他就把艰难的任务交给你了,他真的不想浪费时间,你能帮助他吗?

输入
第1行:小吃街的长度N(2 <= N <= 50000)
第2 - N+1行:N个整数,代表每个店的商品价值 (-10^9 <= S[i] <= 10^9)

输出
czl能买到的最大价值

样例输入
6

-2 11 -4 13 5 -2


样例输出
25


问题思路:本题与普通的最大子段和问题不同的是,最大子段和可以是首尾相接的情况,即可以循环。那么这个题目的最大子段和有两种情况。

(1)正常数组中间的某一段和最大。这个可以通过普通的最大子段和问题求出。

(2)此数组首尾相接的某一段和最大。这种情况是由于数组中间某段和为负值,且绝对值很大导致的,那么我们只需要把中间的和为负值且绝对值最大的这一段序列求出,用总的和减去它就行了。


#include<stdio.h>  
#include<algorithm>  
#include<stdlib.h>  
using namespace std;  
#define  N 55000  
#define inf -99999999  
int a[N],b[N];  
int main()  
{  
    int i,n;  
    long long sum,sum1,sum2,ans; 
    while(scanf("%d",&n)!=EOF)  
    {  
        ans = inf;  
        sum = 0;  
        sum1 = sum2 = 0;  
        for(i = 1; i <= n; i ++)  
        {  
            scanf("%d",&a[i]);  
            b[i] = -a[i];  //数组中所有元素符号取反,再求最大子段和 
            sum += a[i]; //所有元素之和 
        }  
        for(i = 1; i <= n; i ++)  
        {  
            sum1 += a[i]; //最大子段和 
            sum2 += b[i];  
            if(sum1 < 0)  
                sum1 = 0;  
            if(sum2 < 0)  
     		 sum2 = 0;  
            ans = max(ans,sum1);  
            ans = max(ans,sum+sum2);  
        }  
        printf("%lld\n",ans);  
    }  
    return 0;  
}
 
 






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值