欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

[IOI2007] sails 船帆

程序员文章站 2024-01-27 08:23:46
显然答案与杆的顺序无关(按每个高度考虑)。 从高到低考虑杆,设此时的状态为$S$,$S "i]$是高度$i$的帆的数目,初始全为$0$,当前杆的高度为$h$,杆上需要放置的帆的数目为$k$,贪心地,假设两个高度的同等宜选,优先选择更高的;帆尽量放置在$S[i]=0(i\le h)$的高度上,若还有剩 ......

显然答案与杆的顺序无关(按每个高度考虑)。

从高到低考虑杆,设此时的状态为\(s\)\(s[i]\)是高度\(i\)的帆的数目,初始全为\(0\),当前杆的高度为\(h\),杆上需要放置的帆的数目为\(k\),贪心地,假设两个高度的同等宜选,优先选择更高的;帆尽量放置在\(s[i]=0(i\le h)\)的高度上,若还有剩余,设剩下\(t(t\le k)\)个,则放置在除去以选择的高度以外,\(s[i](i\le h)\)\(t\)小的位置。

整理一下,每次选出\(s[1\cdots h]\)中前\(k\)小的(相同大小选下标较大的)高度放置帆。

{5,3} 0 0 1 1 1  +0
{4,3} 1 1 2 1 1  +1
{4,1} 1 2 2 1 1  +1
{3,2} 2 2 3 1 1  +3
{3,2} 3 3 3 1 1  +4
{2,1} 3 4 3 1 1  +3

但是此时\(s\)似乎不好维护。。考虑将过程倒过来,从低到高考虑杆,(\(s\)一开始为空集,其余定义相同),假设两个高度同等宜选,优先选择更低的,其余大致相同。即每次选出\(s[1\cdots h]\)中前\(k\)小的(相同大小选下标较小的)高度放置帆。

{2,1} 1 0        +0
{3,2} 1 1 1      +0
{3,2} 2 2 1      +2
{4,1} 2 2 1 1    +0
{4,3} 3 2 2 2    +4
{5,3} 3 3 3 2 1  +4

这样用平衡树维护\(s\)(初始大小为0,每次长度变化时补0节点),每次查询全局前\(k\)小然后整体加一即可。(省去了下标范围的约束)。进一步可发现,连下标都不用维护了


巧妙的分割线(之前的splay已经弃坑了)

观察各个时态的\(s\),发现它总是一个单调不增的序列,把某时态的\(s\)中相同且相邻分成一块,如图,蓝色方框表示将选出前\(k\)小的位置,注意被完全覆盖的块的高度可以直接区间+1,而割开的块区间+1的范围是反过来的,显然这样的块最多一块,于是可以上线段树来维护。

[IOI2007] sails 船帆

答案也不用每步累加,设\(s[i]\)表示最终状态上高度为\(i\)的帆的数目,显然总答案是\(\sum_i\frac{s[i](s[i]-1)}2\),这与因为从本文第一句照应。

#include <bits/stdc++.h>
#define ll long long 
const int n=1e5+10;

namespace sgt {
    struct sgtnode {
        int mx,mn,add;
    } t[n<<2];
#define ls (x<<1)
#define rs (x<<1|1)
    void update(int x) {
        t[x].mx=std::max(t[ls].mx,t[rs].mx);
        t[x].mn=std::min(t[ls].mn,t[rs].mn);
    }
    void pushr(int x,int add) {t[x].mn+=add,t[x].mx+=add,t[x].add+=add;}
    void pushdown(int x) {
        if(t[x].add) pushr(ls,t[x].add),pushr(rs,t[x].add),t[x].add=0;
    }
    ll calc(int x,int l,int r) {
        if(l==r) return 1ll*t[x].mx*(t[x].mx-1)/2;
        int mid=(l+r)>>1; pushdown(x);
        return calc(ls,l,mid)+calc(rs,mid+1,r);
    }
    void build(int x,int l,int r) {
        t[x].mn=+1e9,t[x].mx=-1e9;
        if(l==r) return; int mid=(l+r)>>1;
        build(ls,l,mid); build(rs,mid+1,r);
    }
    void insert(int x,int l,int r,int p) {
        if(l==r) return void(t[x].mx=t[x].mn=0);
        int mid=(l+r)>>1; pushdown(x);
        if(p<=mid) insert(ls,l,mid,p);
        else insert(rs,mid+1,r,p);
        update(x);
    }
    void modify(int x,int l,int r,int l,int r) {
        if(l>r) return;
        if(l<=l&&r<=r) return pushr(x,1);
        int mid=(l+r)>>1; pushdown(x);
        if(l<=mid) modify(ls,l,mid,l,r);
        if(mid<r) modify(rs,mid+1,r,l,r);
        update(x);
    }
    int getval(int x,int l,int r,int p) {
        if(t[x].mn==t[x].mx) return t[x].mn; 
        int mid=(l+r)>>1; pushdown(x);
        if(p<=mid) return getval(ls,l,mid,p);
        else return getval(rs,mid+1,r,p); 
    }
    int getrangel(int x,int l,int r,int w) {
        if(l==r) return l;
        int mid=(l+r)>>1; pushdown(x);
        if(t[ls].mn<=w) return getrangel(ls,l,mid,w);
        else return getrangel(rs,mid+1,r,w);
    }
    int getranger(int x,int l,int r,int w) {
        if(l==r) return l;
        int mid=(l+r)>>1; pushdown(x);
        if(t[rs].mx>=w) return getranger(rs,mid+1,r,w);
        else return getranger(ls,l,mid,w);
    }
}

int n,m;
std::pair<int,int> a[n];

int main() {
    scanf("%d",&n);
    for(int i=1; i<=n; ++i) {
        scanf("%d%d",&a[i].first,&a[i].second);
        m=std::max(m,a[i].first);
    }
    std::sort(a+1,a+n+1);
    sgt::build(1,1,m);
    for(int i=1,r=0; i<=n; ++i) {
        while(r<a[i].first) sgt::insert(1,1,m,++r);
        int pos=r-a[i].second+1; //被割开的位置 
        int val=sgt::getval(1,1,m,pos);
        int rgl=sgt::getrangel(1,1,m,val);
        int rgr=sgt::getranger(1,1,m,val);
        
//      std::cout<<val<<' '<<rgl<<' '<<rgr<<std::endl;
        
        sgt::modify(1,1,m,rgr+1,r);
        sgt::modify(1,1,m,rgl,rgl+a[i].second-(r-rgr)-1);
    }
    printf("%lld\n",sgt::calc(1,1,m));
    return 0; 
}