题目描述

你有一个m点生命值的奴隶主,奴隶主受伤未死且当前随从数目不超过k则再召唤一个m点生命值的奴隶主。

T次询问,每次询问如果如果对面下出一个n点攻击力的克苏恩,你的英雄期望会受到到多少伤害。

输入

输入第一行包含三个正整数 T,m,k ,T 表示询问组数,m,k 的含义见题目描述。

接下来 T 行,每行包含一个正整数 n ,表示询问进行 n 次攻击后扣减Boss的生命值点数的期望。

输出

输出共 T 行,对于每个询问输出一行一个非负整数,表示该询问的答案对 998244353 取模的结果。

样例输入

3 2 6
1
2
3

样例输出

499122177
415935148
471393168


题解

概率dp+倍增+矩阵乘法

首先需要知道本题弱化版 【bzoj4832】[Lydsy2017年4月月赛]抵制克苏恩 的概率dp写法:不维护期望,只维护概率,统计概率对答案的贡献。设 $p[i][j][k][l]$ 表示 $i$ 回合奴隶主1、2、3血剩余情况为 $j$ 、$k$ 、$l$ 的概率,那么对答案的贡献就是 $\frac {p[i][j][k][l]}{j+k+l+1}$ 。

本题的 $n$ 较大,考虑矩阵乘法。先预处理出状态及转移。然后相当于一个行向量乘以n个方阵,使用快速幂。

但是经过计算可知状态数为 $\sum\limits_{i=0}^kC_{i+m-1}^{m-1}$ ,加上计数器总和最大为166,每次都快速幂复杂度为 $O(T·166^3\log n)$ ,会TLE。

考虑到一个行向量乘以一个方阵的时间时 $O(n^2)$ 的,因此可以倍增预处理出方阵的 $2^i$ 次幂,然后把每个矩阵依次乘到行向量上即可。

时间复杂度 $O(166^3\log n+T·166^2\log n)$

有点卡常。。。

  1. #include <cstdio>
  2. #include <cstring>
  3. #define mod 998244353
  4. typedef long long ll;
  5. int tot = 1;
  6. ll inv(ll x)
  7. {
  8. ll ans = 1 , y = mod - 2;
  9. while(y)
  10. {
  11. if(y & 1) ans = ans * x % mod;
  12. x = x * x % mod , y >>= 1;
  13. }
  14. return ans;
  15. }
  16. struct data
  17. {
  18. ll v[170][170];
  19. data() {memset(v , 0 , sizeof(v));}
  20. ll *operator[](int a) {return v[a];}
  21. data operator*(data &a)
  22. {
  23. data ans;
  24. int i , j , k;
  25. for(i = 1 ; i <= tot ; i ++ )
  26. for(j = 1 ; j <= tot ; j ++ )
  27. for(k = 1 ; k <= tot ; k ++ )
  28. ans[i][j] = (ans[i][j] + v[i][k] * a[k][j]) % mod;
  29. return ans;
  30. }
  31. }A[60];
  32. int f[9][9] , g[9][9][9];
  33. ll ans[170] , tmp[170];
  34. void mul(ll *A , data B)
  35. {
  36. int i , j;
  37. memset(tmp , 0 , sizeof(tmp));
  38. for(i = 1 ; i <= tot ; i ++ )
  39. for(j = 1 ; j <= tot ; j ++ )
  40. tmp[j] = (tmp[j] + A[i] * B[i][j]) % mod;
  41. for(i = 1 ; i <= tot ; i ++ ) A[i] = tmp[i];
  42. }
  43. int main()
  44. {
  45. int T , m , p , i , j , k;
  46. ll n , e;
  47. scanf("%d%d%d" , &T , &m , &p);
  48. A[0][1][1] = A[0][2][1] = A[0][2][2] = 1;
  49. if(m == 1) tot = 3 , A[0][tot][1] = A[0][tot][2] = A[0][tot][3] = inv(2);
  50. else if(m == 2)
  51. {
  52. for(i = 0 ; i <= p ; i ++ )
  53. for(j = 0 ; j <= p ; j ++ )
  54. if(i + j <= p)
  55. f[i][j] = ++tot;
  56. for(i = 0 ; i <= p ; i ++ )
  57. {
  58. for(j = 0 ; j <= p ; j ++ )
  59. {
  60. if(i + j <= p)
  61. {
  62. e = inv(i + j + 1);
  63. A[0][f[i][j]][1] = A[0][f[i][j]][f[i][j]] = e;
  64. if(i) A[0][f[i][j]][f[i - 1][j]] = i * e % mod;
  65. if(j)
  66. {
  67. if(i + j < p) A[0][f[i][j]][f[i + 1][j]] = j * e % mod;
  68. else A[0][f[i][j]][f[i + 1][j - 1]] = j * e % mod;
  69. }
  70. }
  71. }
  72. }
  73. }
  74. else
  75. {
  76. for(i = 0 ; i <= p ; i ++ )
  77. for(j = 0 ; j <= p ; j ++ )
  78. for(k = 0 ; k <= p ; k ++ )
  79. if(i + j + k <= p)
  80. g[i][j][k] = ++tot;
  81. for(i = 0 ; i <= p ; i ++ )
  82. {
  83. for(j = 0 ; j <= p ; j ++ )
  84. {
  85. for(k = 0 ; k <= p ; k ++ )
  86. {
  87. if(i + j + k <= p)
  88. {
  89. e = inv(i + j + k + 1);
  90. A[0][g[i][j][k]][1] = A[0][g[i][j][k]][g[i][j][k]] = e;
  91. if(i) A[0][g[i][j][k]][g[i - 1][j][k]] = i * e % mod;
  92. if(j)
  93. {
  94. if(i + j + k < p) A[0][g[i][j][k]][g[i + 1][j - 1][k + 1]] = j * e % mod;
  95. else A[0][g[i][j][k]][g[i + 1][j - 1][k]] = j * e % mod;
  96. }
  97. if(k)
  98. {
  99. if(i + j + k < p) A[0][g[i][j][k]][g[i][j + 1][k]] = k * e % mod;
  100. else A[0][g[i][j][k]][g[i][j + 1][k - 1]] = k * e % mod;
  101. }
  102. }
  103. }
  104. }
  105. }
  106. }
  107. for(i = 1 ; i < 60 ; i ++ ) A[i] = A[i - 1] * A[i - 1];
  108. while(T -- )
  109. {
  110. scanf("%lld" , &n);
  111. memset(ans , 0 , sizeof(ans));
  112. ans[3] = 1;
  113. for(i = 0 ; i < 60 ; i ++ )
  114. if(n & (1ll << i))
  115. mul(ans , A[i]);
  116. printf("%lld\n" , ans[1]);
  117. }
  118. return 0;
  119. }

【loj2325】「清华集训 2017」小Y和恐怖的奴隶主 概率dp+倍增+矩阵乘法的更多相关文章

  1. LOJ2325. 「清华集训 2017」小 Y 和恐怖的奴隶主【矩阵快速幂优化DP】【倍增优化】

    LINK 思路 首先是考虑怎么设计dp的状态 发现奴隶主的顺序没有影响,只有生命和个数有影响,所以就可以把每个生命值的奴隶主有多少压缩成状态就可以了 然后发现无论是什么时候一个状态到另一个状态的转移都 ...

  2. LOJ2325「清华集训 2017」小Y和恐怖的奴隶主

    题目链接 首先dp很显然,\(f(i,s)\)表示到了第i轮,各种血量人数的情况为s今后的期望攻击boss次数.那么有\(f(i,s)=\frac{1}{num+1}*\sum_{s->s'}( ...

  3. loj #2325. 「清华集训 2017」小Y和恐怖的奴隶主

    #2325. 「清华集训 2017」小Y和恐怖的奴隶主 内存限制:256 MiB时间限制:2000 ms标准输入输出 题目类型:传统评测方式:文本比较   题目描述 "A fight? Co ...

  4. LibreOJ #2325. 「清华集训 2017」小Y和恐怖的奴隶主(矩阵快速幂优化DP)

    哇这题剧毒,卡了好久常数才过T_T 设$f(i,s)$为到第$i$轮攻击,怪物状态为$s$时对boss的期望伤害,$sum$为状态$s$所表示的怪物个数,得到朴素的DP方程$f(i,s)=\sum \ ...

  5. Loj #2324. 「清华集训 2017」小 Y 和二叉树

    Loj #2324. 「清华集训 2017」小 Y 和二叉树 小Y是一个心灵手巧的OIer,她有许多二叉树模型. 小Y的二叉树模型中,每个结点都具有一个编号,小Y把她最喜欢的一个二叉树模型挂在了墙上, ...

  6. [LOJ#2324]「清华集训 2017」小Y和二叉树

    [LOJ#2324]「清华集训 2017」小Y和二叉树 试题描述 小Y是一个心灵手巧的OIer,她有许多二叉树模型. 小Y的二叉树模型中,每个结点都具有一个编号,小Y把她最喜欢的一个二叉树模型挂在了墙 ...

  7. [LOJ#2323]「清华集训 2017」小Y和地铁

    [LOJ#2323]「清华集训 2017」小Y和地铁 试题描述 小Y是一个爱好旅行的OIer.一天,她来到了一个新的城市.由于不熟悉那里的交通系统,她选择了坐地铁. 她发现每条地铁线路可以看成平面上的 ...

  8. 【UOJ#340】【清华集训2017】小 Y 和恐怖的奴隶主(矩阵快速幂,动态规划)

    [UOJ#340][清华集训2017]小 Y 和恐怖的奴隶主(矩阵快速幂,动态规划) 题面 UOJ 洛谷 题解 考虑如何暴力\(dp\). 设\(f[i][a][b][c]\)表示当前到了第\(i\) ...

  9. loj2324 「清华集训 2017」小 Y 和二叉树

    https://loj.ac/problem/2324 太智障,一开始以为中序遍历的第一个点一定是一个叶子,想了个贪心.然而,手算了一下,第一个点都过不了啊. input 5 2 3 4 1 3 3 ...

随机推荐

  1. 深度学习开源库tiny-dnn的使用(MNIST)

    tiny-dnn是一个基于DNN的深度学习开源库,它的License是BSD 3-Clause.之前名字是tiny-cnn是基于CNN的,tiny-dnn与tiny-cnn相关又增加了些新层.此开源库 ...

  2. LVS入门篇(三)之LVS的工作模式和调度算法

    1.NAT模型 (1)原理图: ①.客户端(200.10.10.1)将请求发往前端的负载均衡器(114.100.80.10),请求报文源地址是CIP(客户端IP),后面统称为CIP),目标地址为VIP ...

  3. LVS入门篇(二)之LVS基础

    1. LVS介绍 LVS是Linux虚拟服务器(LinuxVirtualServers),使用负载均衡技术将多台服务器组成一个虚拟服务器.它为适应快速增长的网络访问需求提供了一个负载能力易于扩展,而价 ...

  4. superset 安装测试,基于windows 和 centos7.x

    1.刚开始在windows平台测试搭建,报各种问题,搭建可以参考官网https://superset.incubator.apache.org/installation.html#deeper-sql ...

  5. python的rtree包缺失libspatiaindex.so

    1 准备autoconf工具 yum -y install autoconf automake libtool 2 准备g++编译器 yum -y install gcc gcc-c++ 3 下载并安 ...

  6. 博弈论(Game Theory) - 04 - 纳什均衡

    博弈论(Game Theory) - 04 - 纳什均衡 开始 纳什均衡和最大最小定理是博弈论的两大基石. 博弈不仅仅是对抗,也包括合作和迁就,纳什均衡能够解决这些问题,提供了在数学上一个完美的理论. ...

  7. oracle 查看锁表及解锁的语句

    解锁语句: alter system kill session 'sid, serial#'; alter system kill session '23, 1647'; 查询那些对象被锁: sele ...

  8. 移动端车牌识别/车牌OCR识别

    周末,小编约了朋友商场shopping. 开车进地下车库时,“滴”的一声,完成车牌录入:开车离开时,扫描二维码,输入车牌,完成停车收费.小编不禁感叹科技改变生活,人工智能给生活带来的便利. 车牌自动识 ...

  9. Qt-Qt5最新增加程序图标方式

    亲爱的小伙伴吗,还记得原始的Qt4是怎么给运行程序添加图标的么,是不是先准备一个ICON文件,问了在家里建立一个rc文件,在文件里面加上 IDI_ICON1               ICON    ...

  10. 学习笔记之glog的使用

    下载源码,使用cmake编译,最后得到了32位的静态库 glog.lib 使用库时要注意添加以下预定义: GLOG_NO_ABBREVIATED_SEVERITIES; GOOGLE_GLOG_DLL ...