牛客小白月赛67-E-游戏的买【概率与期望DP】

  • Post author:
  • Post category:其他



链接



传送门



分析


这道题要注意一定要保证游戏能够买到。如何在最优的情况下保证能够买到游戏,正面考虑太复杂了,我们倒着考虑,创建一个数组f用来记录,从第i开始到最后的最优的期望,对于f[n]也就是最后一天,马上就要停止发售了,所以必须要购买,所以这天的期望是0.5a+0.5b,那么对于一般的情况,例如i,我们已经知道了从i+1天开始购买的最佳的期望f[i+1],如何算出从第i天开始的最佳的期望,如果说,第i天不管购买a还是b都是不如之后购买的期望,那么f[i]=f[i+1],如果说第i天不管购买a还是购买b都是比之后购买要更好,那么肯定不会选择再之后购买,所以最好的期望就是0.5a+0.5b,如果说a<f[i+1]< b的话,这个时候我们为了追求利益最大话,有1/2的概率价格为a,还有1/2概率价格为b,当价格b的时候不如后面买的期望,为了使得期望更小我们选择后面买,

注意价格是每天最早就公布了

,所以我们是知道价格然后进行决策的。此时f[i]=0.5a+0.5f[i+1]。以此类推可以得到第一天决策的最佳期望。


实现

#include <bits/stdc++.h>
#define ll long long
#define ls (u << 1)
#define rs (u << 1 | 1)
#define inf 0x3f3f3f3f
#define INF 0x3f3f3f3f3f3f3f3f
using namespace std;
typedef pair<int, int> PII;
const int N = 1e5 + 5;
int a[N], b[N];
double f[N];
void solve() {
	int n;
	scanf("%d", &n);
	for (int i = 1; i <= n; i++) scanf("%d", a + i);
	for (int i = 1; i <= n; i++) scanf("%d", b + i);
	f[n] = 0.5 * a[n] + 0.5 * b[n];
	for (int i = n - 1; i >= 1; i--) {
		if (min(a[i], b[i]) >= f[i + 1]) f[i] = f[i + 1];
		else if (max(a[i], b[i]) < f[i + 1]) f[i] = 0.5 * a[i] + 0.5 * b[i];
		else f[i] = 0.5 * min(a[i], b[i]) + 0.5 * f[i + 1];
	}
	printf("%lf\n", f[1]);
}
int main(){
//	ios::sync_with_stdio(false);
//	cin.tie(0);
	int T = 1;
	cin >> T;
	while (T--) solve();
	return 0;
}



版权声明:本文为gfyy_bkj原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。