题目大意

给你HNOI2015 亚瑟王 - 图1 张牌,顺序进行HNOI2015 亚瑟王 - 图2 轮游戏, 每张牌都有一个触发技能概率,技能都有一个伤害,如果一张牌触发过 以后他都会被跳过,一张牌触发之后这轮结束

题解

因为每一张牌的贡献都是1, 所以我们只需要考虑概率就行。

对于第一张牌, 正难则反,他在HNOI2015 亚瑟王 - 图3 轮中 触发的概率为

HNOI2015 亚瑟王 - 图4%5Er%0A#card=math&code=%5Clarge%201-%281-p%5B1%5D%29%5Er%0A&id=fhS9L)

考虑 第二张牌 如果第一张牌触发过了 那么有一轮 根本就轮不到他了

HNOI2015 亚瑟王 - 图5%5E%7Br-1%7D%0A#card=math&code=%5Clarge%201-%281-p%5B2%5D%29%5E%7Br-1%7D%0A&id=z1TGI)

所以我们考虑用HNOI2015 亚瑟王 - 图6 表示在全部的回合中 前HNOI2015 亚瑟王 - 图7 张牌 触发了HNOI2015 亚瑟王 - 图8 张的概率

考虑转移 当第HNOI2015 亚瑟王 - 图9 张牌发动 因为有HNOI2015 亚瑟王 - 图10 轮都直接跳过了 HNOI2015 亚瑟王 - 图11

HNOI2015 亚瑟王 - 图12)*f%5Bi-1%5D%5Bj-1%5D%0A#card=math&code=%5Clarge%20f%5Bi%5D%5Bj%5D%2B%3D%281-%281-p%5Bi%5D%5E%7Br-j%2B1%7D%29%29%2Af%5Bi-1%5D%5Bj-1%5D%0A&id=o1v6H)

考虑他不发动的情况

HNOI2015 亚瑟王 - 图13*f%5Bi-1%5D%5Bj%5D%0A#card=math&code=%5Clarge%20f%5Bi%5D%5Bj%5D%2B%3D%281-p%5Bi%5D%5E%7Br-j%7D%29%2Af%5Bi-1%5D%5Bj%5D%0A&id=apSZN)

转移的时候 每个底层 加两次

最后收集答案

HNOI2015 亚瑟王 - 图14%7Df%5Bi-1%5D%5Bj%5D*(1-(1-p%5Bi%5D%5E%7Br-j%7D))%0A#card=math&code=%5Clarge%20g%5Bi%5D%3D%5CSigma_0%5E%7Bmin%28r%2Ci-1%29%7Df%5Bi-1%5D%5Bj%5D%2A%281-%281-p%5Bi%5D%5E%7Br-j%7D%29%29%0A&id=FbceP)

  1. #include <cmath>
  2. #include <cstdio>
  3. #include <cstring>
  4. #include <iostream>
  5. #include <algorithm>
  6. using namespace std;
  7. const int N = 234, R = 143;
  8. int n, r, d[N];
  9. double p[N], f[N][R], g[N], pw[N][R];
  10. void work() {
  11. memset(f, 0, sizeof(f)); memset(g, 0, sizeof(g));
  12. int i, j; scanf("%d%d", &n, &r);
  13. for (i = 1; i <= n; i++) scanf("%lf%d", &p[i], &d[i]), pw[i][0] = 1;
  14. if (!r) return (void) puts("0.0000000000");
  15. for (i = 1; i <= n; i++) for (j = 1; j <= r; j++)
  16. pw[i][j] = pw[i][j - 1] * (1.0 - p[i]);
  17. f[1][0] = pw[1][r]; f[1][1] = g[1] = 1.0 - pw[1][r];
  18. for (i = 2; i <= n; i++) for (j = 0; j <= min(i, r); j++) {
  19. if (j) f[i][j] += f[i - 1][j - 1] * (1.0 - pw[i][r - j + 1]);
  20. if (i != j) f[i][j] += f[i - 1][j] * pw[i][r - j];
  21. }
  22. for (i = 2; i <= n; i++) for (j = 0; j <= min(i - 1, r); j++)
  23. g[i] += f[i - 1][j] * (1.0 - pw[i][r - j]);
  24. double ans = 0;
  25. for (i = 1; i <= n; i++) ans += g[i] * d[i];
  26. printf("%.10lf\n", ans);
  27. }
  28. int main() {
  29. int T; cin >> T; while (T--) work();
  30. return 0;
  31. }