• [补题记录] Atcoder Beginner Contest 295(E)


    URL:https://atcoder.jp/contests/abc295

    目录

    E

    Problem/题意

    Thought/思路

    Code/代码


    E

    Problem/题意

    给定长度为 N 的数组 A。进行如下操作:

    • 若 Ai = 0,将 Ai 等概率地变为 1 ~ M 中的任意一个数;
    • 对 A 排序;

    问第 K 个数地期望是多少。

    Thought/思路

    概率 DP。(一开始想不明白这个公式,概率论白雪了)

    设我们要求的 A[k] = x 且 P[i] 为 x = i 的概率,那么就有如下公式:

    E(x) = \sum_{i=1}^{m}i*P(i=x)=\sum_{i=1}^{m}P(x \geqslant i)

     关于这条公式地推导:https://zhuanlan.zhihu.com/p/617048570

    因此接下来的问题就变成了:对于每个 i,求出 P(A[k] >= i)。


    但是我们不知道 A[k] 该怎么取值,所以还需要将 P(A[k] >= i) 转换为:后面 N - K + 1 个数 >= i 的概率,也就是 [K, N] 中的数都 >= i 的概率。(假设已经排好序)

    显然 [K, N] 中的数不会都 >= i,而一般的情况就是:[K, N] 中的前一部分的数 < i、后一部分的数 >= i。


    对于前一部分,我们需要依靠 0 来变成 >= i 的数去替换他们,所以记录前一部分的数的个数为 need,这代表了所需要的 0 的最少数量。

    也就是说,如果 0 的数量(设为 zero)zero < need,那么就永远不可能满足 [K, N] 中的数都 >= i,概率为 0;反之,如果 need <= 0,就一定满足 [K, N] 中的数都 >= i,概率为 1;


    基于概率为 0 的那种情况,就一定能保证 need <= zero。

    而 need 是需要的 0 的最少数量,那么我们就可以设:有 need 个 0 变成了 >= i 的数,其带来的概率为:

    p(need) = C_{zero}^{need} * P^{need} * (1 - P)^{zero-need}

     其中 P = (m - i + 1) / m,意思是:取出 >= i 的数的概率。

    显然一共有 zero 个 0 可以使用,所以考虑 [need, zero] 每一种情况即可。

    Code/代码

    1. #include "bits/stdc++.h"
    2. #define int long long
    3. const int mod = 998244353;
    4. int n, m, k, a[2007], fact[2007], invf[2007];
    5. int ksm(int a, int b) {
    6. int res = 1;
    7. while (b > 0) {
    8. if (b & 1) res = res * a % mod;
    9. a = a * a % mod;
    10. b /= 2;
    11. }
    12. return res;
    13. }
    14. void init() {
    15. fact[0] = 1, invf[0] = ksm(1, mod - 2);
    16. for (int i = 1; i <= 2000; ++ i) {
    17. fact[i] = fact[i - 1] * i % mod;
    18. invf[i] = ksm(fact[i], mod - 2) % mod;
    19. }
    20. }
    21. int C(int x, int y) {
    22. if (x < y) return 0;
    23. return fact[x] * invf[y] % mod * invf[x - y] % mod;
    24. }
    25. signed main() {
    26. std::cin >> n >> m >> k;
    27. for (int i = 1; i <= n; ++ i) std::cin >> a[i];
    28. init();
    29. int ans = 0;
    30. for (int i = 1; i <= m; ++ i) {
    31. int zero = 0, need = n - k + 1;
    32. for (int j = 1; j <= n; ++ j) {
    33. if (a[j] >= i) need --;
    34. if (a[j] == 0) zero ++;
    35. }
    36. if (need <= 0 or need > zero) { // [k, n] 都 >= i,概率为 1;[k, n] 小于 i 的个数,0 补不上,概率为 0。
    37. ans = (ans + (need <= 0 ? 1 : 0)) % mod;
    38. continue;
    39. }
    40. int p1 = (m - i + 1) * ksm(m, mod - 2) % mod; // 选出的数 >= i 的概率 p:(m - i + 1) / m
    41. int p2 = (i - 1) * ksm(m, mod - 2) % mod; // 1 - p:(i - 1) / m
    42. std::vector <int> dp1(zero + 1), dp2(zero + 1);
    43. dp1[0] = dp2[0] = ksm(1, mod - 2);
    44. for (int j = 1; j <= zero; ++ j) {
    45. dp1[j] = dp1[j - 1] * p1 % mod;
    46. dp2[j] = dp2[j - 1] * p2 % mod;
    47. }
    48. // 用 0 补充 >= i 的数
    49. for (int j = need; j <= zero; ++ j) {
    50. ans = (ans + C(zero, j) * dp1[j] % mod * dp2[zero - j] % mod) % mod;
    51. }
    52. }
    53. std::cout << ans;
    54. return 0;
    55. }
  • 相关阅读:
    APS高级计划排程系统,工厂各部门实施前后有哪些区别?
    element ui tree树形控件实现单选操作
    云原生实战课大纲
    linux系统---安装使用nginx
    计算机组成原理 new07 真值和机器数 无符号整数 定点整数 定点小数 $\color{red}{Δ}$
    Oracle与Mysql语法区别
    内网域环境搭建教程
    系统架构设计师-计算机系统基础知识(1)
    有线网卡通过无线网卡使其它设备上网
    事件总结-常用总结
  • 原文地址:https://blog.csdn.net/joyride_run/article/details/133827039