pkusc 快到了……做点题涨涨 rp。

ref我好菜啊QAQ。


可以发现期望只是一个幌子。我们的目的是:对于所有随机的选择方法(一共 \(\binom{2n}{m}\)种),这些选择方法都最优地打出 \(k\) 张牌,他们能造成的伤害的和是多少。

显然的是,能打强化就打强化(不过你好歹也要攻击一张)。记 \(m\) 张卡中分给强化卡的数量为 \(i\)。我们枚举 \(i\),根据 \(i\) 与 \(k\) 的大小关系来决定怎样打牌。

那么 \(i < k\) 时,就打出 \(i\) 张强化卡,\(k-i\) 张攻击卡。(这意味着你能打的强化卡总共才 \(i\) 张,当然是能打强化卡就打强化卡)

\(i \geq k-1\) 时,就打出 \(k-1\) 张强化卡,\(1\) 张攻击卡。(这意味着你能打的强化卡还挺多,留一张攻击就行了)。

记 \(F(i,j)\) 为分给强化卡的数量为 \(i\),打出 \(j\) 张,所有方案翻的倍率的和。\(G(i,j)\) 为分给攻击卡的数量为 \(i\),打出 \(j\) 张,所有方案的(不加强化的)攻击力和。

两者分别对应 \(F(i,i) \times G(m-i,k-i)\) 和 \(F(i,k-1) \times G(m-i,1)\)。为什么可以是这种“和乘和”的形式呢?因为乘法分配律。


现在的问题变成快速计算 \(F\) 和 \(G\)。

关于 \(F\),可以 sort 以后定义一个 \(f\),\(f(i,j)\) 表示(注意是选,不是分)了 \(i\) 张强化牌且最这 \(i\) 张牌中位置靠前的那张牌是所有强化牌中的第 \(j\) 个,这样的所有方案翻的倍率的和。转移看代码。\(G\) 和 \(g\) 也类似。

#include <algorithm>
#include <iostream>
#include <cstring>
#include <cstdio>
using namespace std;
typedef long long ll;
const int mod=998244353;
int T, n, m, k, a[1505], b[1505], c[3005][3005], f[1505][1505];
int g[1505][1505], sum[1505];
int F(int x, int y){
if(x<y) return 0;
if(!y) return c[n][x];
int re=0;
for(int j=x-y+1; j<=n-y+1; j++)
re = (re + (ll)f[y][j]*c[j-1][x-y]%mod) % mod;
//感性理解一下……大概就是把x-y张不打出的牌放到j前头,这里我讲不太清QAQ
return re;
}
int G(int x, int y){
if(x<y) return 0;
int re=0;
for(int j=x-y+1; j<=n-y+1; j++)
re = (re + (ll)g[y][j]*c[j-1][x-y]%mod) % mod;
return re;
}
int main(){
cin>>T;
for(int i=0; i<=3000; i++){
c[i][0] = 1;
for(int j=1; j<=i; j++)
c[i][j] = (c[i-1][j-1] + c[i-1][j]) % mod;
}
while(T--){
memset(f, 0, sizeof(f));
memset(g, 0, sizeof(g));
scanf("%d %d %d", &n, &m, &k);
for(int i=1; i<=n; i++) scanf("%d", &a[i]);
for(int i=1; i<=n; i++) scanf("%d", &b[i]);
sort(a+1, a+1+n);//跟牌的顺序无关,可以sort
sort(b+1, b+1+n);
for(int i=1; i<=n; i++){
f[1][i] = a[i];//初始化f[][],显然只选1张的倍率之和是a[i]
sum[i] = (sum[i-1] + a[i]) % mod;//前缀和,方便转移
}
for(int i=2; i<=n; i++){
for(int j=1; j<=n-i+1; j++)
f[i][j] = (ll)a[j] * (sum[n]-sum[j]+mod) % mod;
//打了i张牌,最前头的是第j张,那它就是f[i-1][j+1..n]的和再乘上第j号元素。这个转移的思想是枚举在打了i-1张牌的时候最前头的是哪一张
for(int j=1; j<=n; j++)
sum[j] = (sum[j-1] + f[i][j]) % mod;
}
for(int i=1; i<=n; i++){
g[1][i] = b[i];
sum[i] = (sum[i-1] + b[i]) % mod;
}
for(int i=2; i<=n; i++){
for(int j=1; j<=n-i+1; j++)
g[i][j] = ((ll)b[j]*c[n-j][i-1]%mod+(sum[n]-sum[j]+mod)%mod) % mod;
//打了i张牌,最前头的是第j张。注意g代表的是(不加强化的)攻击力和。在这种情况下,打了i-1张牌的总情况是c[n-j][i-1]种(j+1..n中选i-1个的方案数),这是第一项;第二项就是继承自g[i-1][j+1..n]
for(int j=1; j<=n; j++)
sum[j] = (sum[j-1] + g[i][j]) % mod;
}
int ans=0;
for(int i=0; i<m; i++)
if(i<k) ans = (ans + (ll)F(i,i)*G(m-i,k-i)%mod) % mod;
else ans = (ans + (ll)F(i,k-1)*G(m-i,1)%mod) % mod;
printf("%d\n", ans);
}
return 0;
}

loj2538 「PKUWC 2018」Slay the Spire的更多相关文章

  1. LOJ #2538. 「PKUWC 2018」Slay the Spire (期望dp)

    Update on 1.5 学了 zhou888 的写法,真是又短又快. 并且空间是 \(O(n)\) 的,速度十分优秀. 题意 LOJ #2538. 「PKUWC 2018」Slay the Spi ...

  2. LOJ #2542. 「PKUWC 2018」随机游走(最值反演 + 树上期望dp + FMT)

    写在这道题前面 : 网上的一些题解都不讲那个系数是怎么推得真的不良心 TAT (不是每个人都有那么厉害啊 , 我好菜啊) 而且 LOJ 过的代码千篇一律 ... 那个系数根本看不出来是什么啊 TAT ...

  3. LOJ #2541. 「PKUWC 2018」猎人杀(容斥 , 期望dp , NTT优化)

    题意 LOJ #2541. 「PKUWC 2018」猎人杀 题解 一道及其巧妙的题 , 参考了一下这位大佬的博客 ... 令 \(\displaystyle A = \sum_{i=1}^{n} w_ ...

  4. LOJ #2540. 「PKUWC 2018」随机算法(概率dp)

    题意 LOJ #2540. 「PKUWC 2018」随机算法 题解 朴素的就是 \(O(n3^n)\) dp 写了一下有 \(50pts\) ... 大概就是每个点有三个状态 , 考虑了但不在独立集中 ...

  5. 【loj2538】 【PKUWC 2018】Slay the Spire dp

    我们不难发现,假设抽了x张攻击牌,y张强化牌,那么肯定是打出尽可能多张的强化牌后,再开始出攻击牌(当然最少要一张攻击牌) 我们设G(i,j)表示:所有(抽到的攻击牌牌数为i,打出的攻击牌牌数为j)的方 ...

  6. LOJ #2537. 「PKUWC 2018」Minimax (线段树合并 优化dp)

    题意 小 \(C\) 有一棵 \(n\) 个结点的有根树,根是 \(1\) 号结点,且每个结点最多有两个子结点. 定义结点 \(x\) 的权值为: 1.若 \(x\) 没有子结点,那么它的权值会在输入 ...

  7. 「PKUWC 2018」Minimax

    传送门:Here 一道线段树合并好题 如果要维护点$ x$的信息,相当于合并$ x$的两棵子树 对于这题显然有:任何叶子节点的权值都可能出现在其祖先上 因而我们只需要在线段树合并的时候维护概率即可 我 ...

  8. loj2540 「PKUWC 2018」随机算法

    pkusc 快到了--做点题涨涨 rp. 记 \(f(S,i)\) 表示 \(S\) 这个集合是决计不能选的(要么属于独立集,要么和独立集相连),或称已经考虑了的,\(i\) 表示此集合对应的最大独立 ...

  9. 「PKUWC 2018」随机算法 (第二版,正解做法)

    上一版貌似是打了 O(3 ^ N) 暴力和 一条链的情况,得了60分.... 第一次做的时候光想练一练暴力...就没去想正解,谁知道正解比暴力好写不知道多少,mmp 设 f(S) 为 选集合S中的点可 ...

随机推荐

  1. [POI2006]TET-Tetris 3D

    题目 二维线段树板子题啊 但是惊讶的发现我不会打标记 毕竟内层是线段树不能\(pushup\)也不能\(pushdown\) 于是考虑一下标记永久化 其实非常显然\(mx_i\)表示区间最大值,\(t ...

  2. 【[GDOI2014]拯救莫莉斯】

    可能我的状态比较鬼畜,应该没有人这么写 设\(dp[i][j][k]\)表示在第\(i\)行,放置油库的状态为\(j\),实际上周围已经有油库或者本身有油库的状态为\(k\)的时候的最小花费 由于我们 ...

  3. robotframwork接口测试(三)—HttpLibrary.HTTP安装

    1. 安装: 1)可以用命令:pip install robotframework-httplibrary(或pip install  --upgrade robotframework-httplib ...

  4. 解决Bug步骤

    1.看报错.将bug定位到文件.类.方法. 2.打断点.将bug定位到具体代码行号. 3.分析断点输出结果. 4.结合报错和断点结果,修改代码. 总之:拆解问题.定位问题.分析问题.解决问题.

  5. C#网络Socket编程

    1.什么是Socket Sockets 是一种进程通信机制,是一个通信链的句柄(其实就是两个程序通信用的) 2.分类 流式套接字(SOCK_STREAM):提供了一种可靠的.面向连接的双向数据传输服务 ...

  6. PAT——1019. 数字黑洞

    给定任一个各位数字不完全相同的4位正整数,如果我们先把4个数字按非递增排序,再按非递减排序,然后用第1个数字减第2个数字,将得到一个新的数字.一直重复这样做,我们很快会停在有“数字黑洞”之称的6174 ...

  7. 调试libRTMP代码来分析RTMP协议

    RTMP是Real Time Messaging Protocol(实时消息传输协议)的首字母缩写.该协议基于TCP,是一个协议族,常用在视频直播领域.RTMP协议的默认端口是1935. 学习一个协议 ...

  8. 使用Docker发布.NET CORE API

    1.使用VS 2015 新建了一个Core API项目,然后把他的依赖升级到最新(我机器VS 2015默认的包都是rc版本),然后publish. 2.在publish目录的同级目录下,新建Docke ...

  9. Java I/O系列(一)InputStream与OutputStream源码分析及理解

    1. InputStream 定义 字节输入流,是一个抽象类,核心是通过read()方法,从数据源中读取一个个字节出来,另有skip,mark功能 核心源码理解 源码: public abstract ...

  10. Web—13-判断网站请求来自手机还是pc浏览器

    判断网站请求来自手机还是pc浏览器 #判断网站来自mobile还是pc def checkMobile(request): """ demo : @app.route(' ...