类型:DP
题目:http://acm.hdu.edu.cn/showproblem.php?pid=3203
来源: 2009 Shanghai Invitation Contest Host by DHU思路:设状态dp(i, 0)表示第i个人来之前门坏,到结束时的最小花费,dp(i, 1)表示第i个人来之前门好,到结束时的最小花费。
当第i个人来之前门坏时,YY有两种选择,修或者不修,得到两种状态,dp(i + 1, 0) + b 和 a + p * dp(i + 1, 0) + (1.0 - p) * dp(i + 1, 1)
当第i个人来之前门好时,只有一种情况[按概率是否踢坏门],p * dp(i + 1, 0) + (1.0 - p) * dp(i + 1, 1)
初始化:f[n][0] = min(a, b), f[n][1] = 0
逆推求解
参考: http://www.cnblogs.com/zcwwzdjn/archive/2012/02/25/2367364.html// hdoj 3203 Door Repairing
// ac 46MS 1796K
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <algorithm>
using namespace std;
const int MAXN = 100010;
int n, d, a, b, i;
double p, f[MAXN][2];
int main() {
while (scanf("%d %d %d %d", &n, &d, &a, &b) != EOF, n || d || a || b) {
p = d / 100.0;
if (n <= 1)
printf("%.4lf\n", 0.0);
else {
f[n][0] = min(a, b), f[n][1] = 0;
for (i = n - 1; i >= 1; i --) {
f[i][0] = min(f[i + 1][0] + b, a + p * f[i + 1][0] + (1.0 - p) * f[i + 1][1]);
f[i][1] = p * f[i + 1][0] + (1.0 - p) * f[i + 1][1];
}
printf("%.4lf\n", f[1][1]);
}
}
return 0;
}