【arc073e】Ball Coloring(线段树,贪心)

【arc073e】Ball Coloring(线段树,贪心)

题面

AtCoder
洛谷

题解

大型翻车现场,菊队完美压中男神的模拟题
首先钦定全局最小值为红色,剩下的袋子按照其中较大值排序。
枚举前面连续的一段是什么颜色,那么此时我们就知道了两种颜色的\(max\),那么只需要考虑蓝色的\(min\)就好了。
答案拆开后变成了\(R_{max}*B_{max}+R_{min}*B_{min}-R_{min}B_{max}-R_{max}B_{min}\)
此时前三项已知,只需要求最后一项的最大值。
既然要\(B_{min}\)最大的话,那么显然按照较大值排序之后,如果除包含最小值的那一袋的最大值染红,那么前面一段一定选择红色,后面一段一定选择蓝色。
否则必定是前面一段蓝色,中间一个红色,后面又是一段蓝色。
那么线段树维护一下就好了。然而直接维护前缀和后缀最大最小值就好了

写错一堆特判调半天

#include<iostream>
#include<cstdio>
#include<algorithm>
using namespace std;
#define ll long long
#define MAX 200200
#define lson (now<<1)
#define rson (now<<1|1)
const int inf=1e9;
inline int read()
{
    int x=0;bool t=false;char ch=getchar();
    while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
    if(ch=='-')t=true,ch=getchar();
    while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
    return t?-x:x;
}
int n,Rmin,Rmax,Bmin,Bmax;ll ans=2e18;
struct Node{int x,y;}a[MAX];
bool operator<(Node a,Node b){if(a.x!=b.x)return a.x>b.x;return a.y>b.y;}
struct SegMentTree
{
    int t1[MAX<<2],t2[MAX<<2],a[MAX];
    void pushup(int now){t1[now]=max(t1[lson],t1[rson]);t2[now]=min(t2[lson],t2[rson]);}
    void Build(int now,int l,int r)
    {
        if(l==r){t1[now]=t2[now]=a[l];return;}
        int mid=(l+r)>>1;
        Build(lson,l,mid);Build(rson,mid+1,r);
        pushup(now);
    }
    int Querymx(int now,int l,int r,int L,int R)
    {
        if(L<=l&&r<=R)return t1[now];
        int mid=(l+r)>>1,ret=0;
        if(L<=mid)ret=max(ret,Querymx(lson,l,mid,L,R));
        if(R>mid)ret=max(ret,Querymx(rson,mid+1,r,L,R));
        return ret;
    }
    int Querymn(int now,int l,int r,int L,int R)
    {
        if(L<=l&&r<=R)return t2[now];
        int mid=(l+r)>>1,ret=1e9;
        if(L<=mid)ret=min(ret,Querymn(lson,l,mid,L,R));
        if(R>mid)ret=min(ret,Querymn(rson,mid+1,r,L,R));
        return ret;
    }
}T;
int main()
{
    n=read();int mn=1e9;
    for(int i=1;i<=n;++i)a[i].x=read(),a[i].y=read();
    for(int i=1;i<=n;++i)if(a[i].x<a[i].y)swap(a[i].x,a[i].y);
    for(int i=1;i<=n;++i)mn=min(mn,a[i].y);
    for(int i=1;i<=n;++i)if(a[i].y==mn){swap(a[i],a[n]);break;}
    Rmin=mn;sort(&a[1],&a[n]);
    for(int i=1;i<n;++i)T.a[i]=a[i].y;
    T.Build(1,1,n-1);
    for(int i=1;i<n;++i)
    {
        Rmax=a[1].x;
        Bmax=max(a[n].x,max(T.Querymx(1,1,n-1,1,i),a[i+1].x));
        Bmin=min(a[n].x,min(T.Querymn(1,1,n-1,1,i),i==n-1?inf:a[n-1].x));
        ans=min(ans,1ll*(Rmax-Rmin)*(Bmax-Bmin));
    }
    for(int i=1;i<n;++i)
    {
        Bmax=max(a[n].x,a[1].x);
        Rmax=max(T.Querymx(1,1,n-1,1,i),i==n-1?0:max(T.Querymx(1,1,n-1,i+1,n-1),a[i+1].x));
        Bmin=min(a[n].x,min(i==n-1?inf:a[i+1].y,i==n-2?inf:a[n-1].x));
        ans=min(ans,1ll*(Rmax-Rmin)*(Bmax-Bmin));
    }
    printf("%lld\n",ans);
    return 0;
}

猜你喜欢

转载自www.cnblogs.com/cjyyb/p/10472978.html