51nod1065 最小正子段和

51nod1065 最小正子段和

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<algorithm>
#include<cstring>
#include<string>
#include<set>
using namespace std;
typedef long long ll;
ll a[50010];
struct node
{
    ll sum,no;
};
node s[50010],t[50010];
bool cmp(node aa,node bb)
{
    if(aa.sum==bb.sum) return aa.no<bb.no;
    return aa.sum<bb.sum;
}
int main()
{
   // freopen("out.txt","w",stdout);
    int n;
    ll ans=99999999999999;
    scanf("%d",&n);
    for(int i = 1; i <= n; ++i) {
        scanf("%lld",&a[i]);
        s[i].sum=a[i]+s[i-1].sum;
        s[i].no=i;
    }
    s[0].sum=0;
    s[0].no=0;
    sort(s,s+n+1,cmp);
    int o = 0;
    for(int i = 0; i <= n; ++i) {
        if(s[i].sum!=s[i-1].sum) {
            t[o].sum=s[i].sum;
            t[o++].no=s[i].no;
        }
    }
    for(int i = 1; i <= o; ++i) {
        if(t[i-1].no<t[i].no) {
            ans=min(ans,t[i].sum-t[i-1].sum);
        }
    }
    cout<<ans<<endl;
    return 0;
}

夹克爷的解答链接

按照前n项和大小排序,将相同的项合并
那么答案一定是相邻的两个
设ABC是排序后的结果,如果A同B不能组成序列,而A同C可以组成序列,那么B同C也可以组成序列,并且BC会是一个更优的解。

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值