HDU 2059 龜兔賽跑
Problem Description
據說在很久很久以前,可憐的兔子經曆了人生中最大的打擊——賽跑輸給烏龜後,心中郁悶,發誓要報仇雪恨,于是躲進了杭州下沙某農業園卧薪嘗膽潛心修煉,終于練成了絕技,能夠毫不休息得以恒定的速度(VR m/s)一直跑。兔子一直想找機會好好得教訓一下烏龜,以雪前恥。
最近正值HDU舉辦50周年校慶,社會各大名流齊聚下沙,兔子也趁此機會向烏龜發起挑戰。雖然烏龜深知獲勝希望不大,不過迫于輿論壓力,隻能接受挑戰。
比賽是設在一條筆直的道路上,長度為L米,規則很簡單,誰先到達終點誰就算獲勝。
無奈烏龜自從上次獲勝以後,成了名龜,被一些八卦雜志稱為“動物界的劉翔”,廣告不斷,手頭也有了不少積蓄。為了能夠再赢兔子,烏龜不惜花下血本買了最先進的武器——“”小飛鴿”牌電動車。這輛車在有電的情況下能夠以VT1 m/s的速度“飛馳”,可惜電池容量有限,每次充滿電最多隻能行駛C米的距離,以後就隻能用腳來蹬了,烏龜用腳蹬時的速度為VT2 m/s。更過分的是,烏龜竟然在跑道上修建了很多很多(N個)的供電站,供自己給電動車充電。其中,每次充電需要花費T秒鐘的時間。當然,烏龜經過一個充電站的時候可以選擇去或不去充電。
比賽馬上開始了,兔子和帶着充滿電的電動車的烏龜并列站在起跑線上。你的任務就是寫個程式,判斷烏龜用最佳的方案進軍時,能不能赢了一直以恒定速度奔跑的兔子。
Input
本題目包含多組測試,請處理到檔案結束。每個測試包括四行:
第一行是一個整數L代表跑道的總長度
第二行包含三個整數N,C,T,分别表示充電站的個數,電動車沖滿電以後能行駛的距離以及每次充電所需要的時間
第三行也是三個整數VR,VT1,VT2,分别表示兔子跑步的速度,烏龜開電動車的速度,烏龜腳蹬電動車的速度
。。。。。
Output
當烏龜有可能赢的時候輸出一行 “What a pity rabbit!”。否則輸出一行”Good job,rabbit!”;
題目資料保證不會出現烏龜和兔子同時到達的情況。
Sample Input
100
3 20 5
5 8 2
10 40 60
100
3 60 5
5 8 2
10 40 60
Sample Output
Good job,rabbit!
What a pity rabbit!
這道題也是不錯,一開始想一步一步來,但是這是不一定的,前一步的最優解并不一定是下一步的依據,其中有剩餘電量,充不充之類的雲雲,但是要動歸又不知道怎麼歸。。。。 也是參考了别人的。。。 這樣是不對的。。
應該是這個點i他前面的點 從0到 它之前即i-1 的dp值以及從這些點充滿電到這點時間的和的最小值。至于為什麼每個都假設充滿電呢,因為對一個點有兩種情況,有剩餘電量繼續走不沖,或是沖了再走,但是對于第一種情況,可以認為,他之前點(如i-4~i的轉移)的資料已經把他不沖給包括進去了,是以再考慮的話就重複了,就是這樣。
應該先找到動态規劃的對象,再根據特性來簡化公式,學習了。注意細節,double啊,第一個0點不用T之類的。
代碼
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std;
int a[];
double dp[];
int main()
{
int L;
while(scanf("%d",&L)!=EOF)
{
int N,C,T;
int VR,VT1,VT2;
scanf("%d %d %d",&N,&C,&T);
scanf("%d %d %d",&VR,&VT1,&VT2);
memset(a,,sizeof(a));
memset(dp,,sizeof(dp));
for(int i=; i<=N; i++)
scanf("%d",&a[i]);
a[N+]=L;
double temp;
for(int i=; i<=N+; i++)
{
double min1=;
for(int j=; j<i; j++)
{
int k=a[i]-a[j];
if(k<C)
temp=k*/VT1;
else
temp=(k-C)*/VT2+C*/VT1;
if(j!=)
temp+=T;
if(min1>dp[j]+temp)
min1=dp[j]+temp;
}
dp[i]=min1;
}
printf(dp[N+]>(L*/VR)?"Good job,rabbit!\n":"What a pity rabbit!\n");
}
return ;
}