【BZOJ1293】【SCOI2009】生日礼物 单调性

链接:

#include <stdio.h>
int main()
{
    puts("转载请注明出处[辗转山河弋流歌 by 空灰冰魂]谢谢");
    puts("网址:blog.csdn.net/vmurder/article/details/46376019");
}

题解:

首先我们把所有元素排一下序,然后枚举最小值,那么最大值是非严格单调上升的,就是一个珠子换成其后第一个的同颜色珠子时,将更新一下最大值,而最小珠子则刚好是其后第一个(反之则有空下来的永远用不上的珠子,不合逻辑。。2333)
结束。

代码:

狂野的long long 和开大数组啊……

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define N 3001000
#define M 6500
#define inf 0x3f3f3f3f3f3f3f3fll
using namespace std;
struct Eli
{
    int i;
    long long x;
    bool operator < (const Eli &A)const
    {return x<A.x;}
}s[N];
int crs[N];
int n,m,cnt;
int head[M],tail[M];
int main()
{
// freopen("test.in","r",stdin);

    int i,k;

    scanf("%d%d",&n,&m);
    for(i=1;i<=m;i++)
    {
        scanf("%d",&k);
        head[i]=cnt+1;
        while(k--)
        {
            scanf("%d",&s[++cnt].x);
            s[cnt].i=cnt;
        }
        tail[i]=cnt;
    }
    sort(s+1,s+n+1);
    for(i=1;i<=n;i++)crs[s[i].i]=i;

    long long um=inf,mx=0;
    for(i=1;i<=m;i++)
    {
        um=min(um,s[crs[tail[i]]].x);
        mx=max(mx,s[crs[head[i]]].x);
    }

    long long ans=mx-s[1].x;
    for(i=1;s[i].x<um;i++)
    {
        mx=max(mx,s[crs[s[i].i+1]].x);
        ans=min(ans,mx-s[i+1].x);
    }
    printf("%lld\n",ans);

    return 0;
}

你可能感兴趣的:(生日礼物,单调性,BZOJ1293,SCOI2009)