2017.07.15【NOIP提高組】模擬賽B組 積木(brick) 題解

原題:

http://172.16.0.132/senior/#contest/show/2061/3

題目描述:

小A正在搭積木。有N個位置可以讓小A使用,初始高度都為0。小A每次搭積木的時候,都會選定一個擁有相同高度的區(qū)間[A..B],然后將位置[A+1..B-1]上的所有積木的高度加一。不幸的是,小A把積木搭好之后沒多久,小A調(diào)皮的弟弟就將其中若干個位置上的積木弄倒了。小A想知道他原來的積木是如何擺放的,所以他求助于你,請你告訴他原來有多少種可能的擺法。


RTqz6d8PGa7fIusnDHuXomatF4r61BcdrzHMStX+blNXX9SbmvKQi26HoDWRAEQbi7ETEQBEEQRAwEQRAEEQNBEAQBRAwEQRAEEQNBEARBxEAQBEEAEQNBEARBxEAQBEEQMRAEQRBAxEAQBEEQMRAEQRDM5v8HUTJOPe6lssUAAAAASUVORK5CYII=.png

輸入:

第一行為一個正整數(shù)N,表示小A有N個位置。
第二行有N個由空格分隔的整數(shù)Hi,表示第i個位置的積木高度。-1表示這個位置上的積木已經(jīng)被弄倒了

輸出:

唯一的一行,輸出包括可能的擺法mod 1,000,000,007的結(jié)果。

樣例輸入:

輸入1:
3
-1 2 -1
輸入2:
-1 -1 -1
輸入3:
6
-1 -1 -1 2 -1 -1

樣例輸出:

輸出1:
0
輸出2:
2
輸出3:
3

數(shù)據(jù)范圍限制:

對于50%的數(shù)據(jù) 1<=N<=1000 -1<=Hi<=1000
對于80%的數(shù)據(jù) 1<=N<=10000
對于100%的數(shù)據(jù) 1<=N<=20000 -1<=Hi<=10000

分析:

對于每一個位置,求出這一個位置的極高,例如對于7而言,每一個位置的極高為0,1,2,3,2,1,0,對于8而言,每一個位置的極高為0,1,2,3,3,2,1,0;
設f[i][j]表示到第i個位置,高度為j的方案數(shù),如果當前位置被推倒,則f[i][j]=f[i-1][j-1]+f[i-1][j]+f[i-1][j+1];對于知道當前位置的高度時,上式所有的j=a[i];注意用滾動dp做,否則數(shù)組會炸!!!
這樣的時間復雜度是O(n^2)
優(yōu)化
在所有程序中,取mod運算時很慢的,所以盡量減少取mod運算,每10次取1次mod即可

實現(xiàn)

#include<cstdio>
#include<cstring>

int tt,t,n,i,j,a[20001];
long long f[2][10001];
int h(int x)
{
    if(x<n/2+1) return x-1;
    return n-x;
}
int main()
{
    freopen("brick.in","r",stdin);freopen("brick.out","w",stdout);
    scanf("%d",&n);
    for(i=1;i<=n;i++) scanf("%d",&a[i]);
    f[0][0]=1;
    for(i=1;i<=n;i++)
    {
        t=h(i);
        tt=h(i-1);
        if(a[i]!=-1)
        {
            if(a[i]<=t)
            {
                f[1][a[i]]=f[1][a[i]]+f[0][a[i]];
                if(a[i]>=1) f[1][a[i]]=f[1][a[i]]+f[0][a[i]-1];
                                if(a[i]+1<=tt) f[1][a[i]]=f[1][a[i]]+f[0][a[i]+1];
                if(i%10==0) f[1][a[i]]%=1000000007;
            }
        }
        else
        {
            for(j=0;j<=t;j++) 
            {
                f[1][j]=f[1][j]+f[0][j];
                if(j>=1) f[1][j]=f[1][j]+f[0][j-1];
                if(j+1<=tt) f[1][j]=f[1][j]+f[0][j+1];
                if(i%10==0) f[1][j]%=1000000007;
            }
        }
        memcpy(f[0],f[1],sizeof(f[1]));
        memset(f[1],0,sizeof(f[1]));
    }
    printf("%lld",f[0][0]%1000000007);
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,572評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,071評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,409評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,569評論 1 307
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,360評論 6 404
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 54,895評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 42,979評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,123評論 0 286
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,643評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,559評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,742評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,250評論 5 356
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 43,981評論 3 346
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,363評論 0 25
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,622評論 1 280
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,354評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,707評論 2 370

推薦閱讀更多精彩內(nèi)容