【NOIP2017】跳房子

這題我0分。
比賽時,我一眼出正解,哈哈,太水了!
這題不就是一個二分+DP+單調隊列嗎?
然而,細節決定成敗。
我錯了許多細節,就掛了。
我只考了0分。。。
首先,這題知足一個條件:
保證g變大後,若是原來知足條件,如今也會知足條件;而若是原來不知足條件,如今就有可能知足條件。
g變小後,若是原來知足條件,如今不必定會知足條件;而若是原來不知足條件,如今就必定不可能知足條件。
因此,咱們能夠用二分找出最合適的g的值。
已知,\({0\leq g\leq 10^9}\),且跳躍的範圍是 Max(1,d-g) ~ d+g。
咱們就能夠列出狀態轉移方程了:\[{F_i=max(F_j)+S_i\space\space\space(X_j+a\leq X_i \bigwedge X_j+b\geq X_i)}\]
其中,a爲跳躍最短距離,b爲跳躍的最長距離。
這樣作的時間複雜度是\({O(log_210^9n^2)}\),很明顯會時超50分。
因此,咱們就要把DP優化一下了。
咱們很容易發現,狀態轉移方程中,對於不一樣的i,\({max(F_j)}\)的值多是同樣的,但咱們的程序卻會從一個較大的區間 上一次最後一個 找到的位置+1 ~ i-1 中找 ,這就是程序中最耗時的地方。
怎麼優化呢?
咱們有多種優化方式,其中我推薦兩種:大根堆,還有單調隊列。大根堆碼量大,而單調隊列方便快捷,所以我比較喜歡用單調隊列。
\({queue_i}\)表示單調隊列的第i個元素,用head表示單調隊列中有效範圍內的第一個元素的下標,用tail表示單調隊列中有效範圍內的最後一個元素的下標。
單調隊列存的是元素的下標,即\({F_i}\)\({X_i}\)中的 i ,這樣能方便判斷。
因爲這個單調隊列是遞減的(即第一個元素最大,第二個元素比第一個小,第三個比第二個小……最後一個是最小的),因此咱們每次使用的最大值就是單調隊列中有效範圍內的第一個元素對應的值。
那麼咱們的狀態轉移方程就能夠變成這個樣子了:\[{F_i=F_{queue_{head}}+S_i\space\space(X_{queue_{head}}+a \leq X_i \bigwedge X_{queue_{head}}+b \geq X_i)}\]優化

其中,a爲跳躍最短距離,b爲跳躍的最長距離。
這樣用起來是很方便的,可是,重點來了——
怎樣才能保持單調隊列的單調性(使單調隊列遞減)和有效性(使\({(X_{queue_{head}}+a \leq X_i \bigwedge X_{queue_{head}}+b \geq X_i)}\))呢?
首先,咱們每一次加入元素時,若是 (new是新加入的元素),也就是說這個樣子(越高的值越大):
spa

許多人都會認爲要變成下面這個樣子:
3d

第i個柱子上面的數字是X[queue[i]]的值。
因爲新加入單調隊列的數,都是能夠跳到第i個格子上的,即\({X_{new}+a\leq X_i\bigwedge X_{new}+b\geq X_i}\)
而X又是遞增的,因此\({X_{new}+a\leq X_i\leq X_{i+1}\leq X_{i+2}\cdots X_n}\)
可是從5(\({X_{new}}\))這個位置出發,能跳到的最遠距離絕對比4遠,因此當5不能跳到某一個地方時,4也絕對跳不到那個位置。因此4就沒用了。
所以咱們能夠把4刪掉(即tail-1),最後再把5加入,變成下面這個樣子:
code

有時候咱們要刪除不少元素,以下面這個例子:

變成blog

咱們就要用一個while循環來刪除F值小於等於F[new]的數。隊列


但咱們的queue[head]是會過時的(queue[head]跳不到第i個格子),這時咱們的queue[head]就不能用了。
咱們要刪掉queue[head],怎麼刪掉呢?直接head+1就行了。
最後一點,建議同窗們把不能到達的點的F賦值爲-maxlongint!io


#include<cstdio>
using namespace std;
#define maxlongint 1999999999
int f[500001],queue[500001],x[500001],s[500001];
int main()
{
    freopen("jump.in","r",stdin);
    freopen("jump.out","w",stdout);
    int n,d,k,l=0,r=1000000000,mid,i,j,t,ans=-1,maxx,minn,head,tail,last;
    bool bk,bz;
    scanf("%d%d%d",&n,&d,&k);
    for(i=1;i<=n;i++) scanf("%d%d",&x[i],&s[i]);
    l=0;r=1000000000;
    while(l<=r)
    {
        mid=(l+r)/2;
        maxx=d+mid;bk=false;bz=true;
        minn=d-mid;last=0;
        if(minn<1) minn=1;
        tail=head=1;
        queue[1]=0;
        for(i=1;i<=n;i++)
        {
            if(maxx>=x[i]&&minn<=x[i])
            {
                bz=false;
                break;
            }
        }
        if(bz)
        {
            l=mid+1;
            continue;
        }
        for(i=1;i<=n;i++)
        {
            f[i]=maxlongint;
            for(j=last+1;j<i;j++)
            {
                if(x[j]+minn>x[i]) break;
                if(x[j]+maxx<x[i]) continue;
                last=j;
                if(f[j]==maxlongint) continue;
                while(head<=tail&&f[queue[tail]]<=f[j]) queue[tail--]=0;
                queue[++tail]=j;
            }
            while(head<tail&&x[queue[head]]+maxx<x[i]) head++;
            if(x[queue[head]]+maxx<x[i]||x[queue[head]]+minn>x[i]) f[i]=maxlongint;
            else f[i]=f[queue[head]]+s[i];
            if(f[i]<maxlongint&&f[i]>=k)
            {
                bk=true;
                break;
            }
        }
        if(bk)
        {
            ans=mid;
            r=mid-1;
        }
        else l=mid+1;
    }
    printf("%d\n",ans);
    return 0;
}
相關文章
相關標籤/搜索