题目链接

LOJ:https://loj.ac/problem/2538

Solution

计数好题。

首先可以发现这题和期望没关系。

其次对于手上的一套牌,设我们有\(a\)张强化牌,那么:

  • 如果\(a\geqslant k-1\),那么我们显然是从大到小打出\(k-1\)张强化牌,最后打出一张最大的攻击牌。
  • \(\rm otherwise\),我们打出所有的强化牌,再从大到小打出攻击牌。

那么就可以\(dp\)了。

对于强化牌,我们从大到小排序,设\(f[i][j]\)表示当前考虑了前\(i\)种牌,打出了\(j\)种,所有方案的倍率之和。

那么可以得到转移:

  • \(j\leqslant k-1\),我们显然打出这张牌是最优的,\(f[i][j]=f[i-1][j]+f[i-1][j-1]\cdot w[i]\)。
  • \(\rm otherwise\),选或不选这张牌我们都不打出,\(f[i][j]=f[i-1][j]+f[i][j]\)。

对于攻击牌,我们从小到大排序,设\(g[i][j]\)表示当前考虑了前\(i\)种牌,打出了\(j\)种,所有方案的伤害之和。

  • \(j\leqslant m-(k-1)\),此时我们只能打出一张牌,\(g[i][j]=g[i-1][j]+\binom{i-1}{j-1}\cdot w[i]\)。
  • \(\rm otherwise\),我们可以打出多张牌,且应该尽量打后面的牌,\(g[i][j]=g[i-1][j-1]+g[i-1][j]+\binom{i-1}{j-1}\cdot w[i]\)。

第一位可以逆循环然后去掉。

最后答案就是\(ans=\sum_{i=0}^m f[i]g[m-i]\)。

#include<bits/stdc++.h>
using namespace std; void read(int &x) {
x=0;int f=1;char ch=getchar();
for(;!isdigit(ch);ch=getchar()) if(ch=='-') f=-f;
for(;isdigit(ch);ch=getchar()) x=x*10+ch-'0';x*=f;
} void print(int x) {
if(x<0) putchar('-'),x=-x;
if(!x) return ;print(x/10),putchar(x%10+48);
}
void write(int x) {if(!x) putchar('0');else print(x);putchar('\n');} #define lf double
#define ll long long const int maxn = 2e5+10;
const int inf = 1e9;
const lf eps = 1e-8;
const int mod = 998244353; int add(int x,int y) {return x+y>mod?x+y-mod:x+y;}
int del(int x,int y) {return x-y<0?x-y+mod:x-y;}
int mul(int x,int y) {return 1ll*x*y-1ll*x*y/mod*mod;} int n,m,k,a[maxn],b[maxn],f[maxn],g[maxn],fac[maxn],ifac[maxn],inv[maxn]; void prepare() {
inv[0]=inv[1]=fac[0]=ifac[0]=1;
for(int i=2;i<=3000;i++) inv[i]=mul(mod-mod/i,inv[mod%i]);
for(int i=1;i<=3000;i++) fac[i]=mul(fac[i-1],i);
for(int i=1;i<=3000;i++) ifac[i]=mul(ifac[i-1],inv[i]);
} int c(int x,int y) {return x>=y?mul(fac[x],mul(ifac[y],ifac[x-y])):0;} void solve() {
memset(f,0,sizeof f);
memset(g,0,sizeof g);
read(n),read(m),read(k);
for(int i=1;i<=n;i++) read(a[i]);
for(int i=1;i<=n;i++) read(b[i]);
sort(a+1,a+n+1,greater<int> ());
f[0]=1;
for(int i=1;i<=n;i++)
for(int j=n;j;j--)
if(j<=k-1) f[j]=add(f[j],mul(f[j-1],a[i]));
else f[j]=add(f[j],f[j-1]);
sort(b+1,b+n+1);
for(int i=1;i<=n;i++)
for(int j=n;j;j--)
if(j<=m-k+1) g[j]=add(g[j],mul(c(i-1,j-1),b[i]));
else g[j]=add(g[j],add(g[j-1],mul(c(i-1,j-1),b[i])));
int ans=0;
for(int i=0;i<=m;i++) ans=add(ans,mul(f[i],g[m-i]));
write(ans);
} int main() {
prepare();
int t;read(t);while(t--) solve();
return 0;
}

[LOJ2538] [PKUWC2018] Slay the Spire的更多相关文章

  1. LOJ2538 PKUWC2018 Slay the Spire DP

    传送门 不想放题面了,咕咕咕咕咕 这个期望明明是用来吓人的,其实要算的就是所有方案的最多伤害的和. 首先可以知道的是,能出强化牌就出强化牌(当然最后要留一张攻击牌出出去),且数字尽量大 所以说在强化牌 ...

  2. [LOJ2538][PKUWC2018]Slay the Spire:DP

    分析 学会新姿势!我们可以通过调整DP顺序来体现选取物品的优先顺序! 显然选取强化牌的最优策略是倍数从高到低,能选就选,最多选\(k-1\)张,选取攻击牌的最优策略是伤害从高到低,尽量少选,但最少选\ ...

  3. 题解-PKUWC2018 Slay the Spire

    Problem loj2538 Solution 在考场上当然要学会写暴力,考虑如果手上已经有了\(a\)张攻击牌和\(b\)张强化牌: 首先强化牌会在攻击牌之前用(废话),其次要将两种牌分别从大往小 ...

  4. BZOJ.5467.[PKUWC2018]Slay the Spire(DP)

    LOJ BZOJ 洛谷 哪张能力牌能乘攻击啊,太nb了叭 显然如果有能力牌,那么应该选最大的尽可能的打出\(k-1\)张. 然后下面说的期望都是乘总方案数后的,即所有情况的和.然后\(w_i\)统一用 ...

  5. [PKUWC2018] Slay the spire

    Description 现在有 \(n\) 张强化牌和 \(n\) 张攻击牌: 攻击牌:打出后对对方造成等于牌上的数字的伤害. 强化牌:打出后,假设该强化牌上的数字为 \(x\),则其他剩下的攻击牌的 ...

  6. 【洛谷5299】[PKUWC2018] Slay the Spire(组合数学)

    点此看题面 大致题意: 有\(n\)张强化牌\(a_i\)和\(n\)张攻击牌\(b_i\),每张牌有一个权值(强化牌的权值大于\(1\)),每张强化牌能使所有攻击牌的权值乘上这张强化牌的权值,每张攻 ...

  7. 洛谷 P5299 - [PKUWC2018]Slay the Spire(组合数学+dp)

    题面传送门 hot tea 啊--这种风格及难度的题放在省选 D2T1 左右还是挺喜闻乐见的罢 首先考虑对于固定的 \(m\) 张牌怎样求出最优的打牌策略,假设我们抽到了 \(p\) 张强化牌,攻击力 ...

  8. loj #2538. 「PKUWC2018」Slay the Spire

    $ \color{#0066ff}{ 题目描述 }$ 九条可怜在玩一个很好玩的策略游戏:Slay the Spire,一开始九条可怜的卡组里有 \(2n\) 张牌,每张牌上都写着一个数字\(w_i\) ...

  9. BZOJ 5467 Slay the Spire

    BZOJ 5467 Slay the Spire 我的概率基础也太差了.jpg 大概就是这样,因为强化牌至少翻倍,所以打出的牌必定是全部的强化牌或者$k-1$个强化牌,然后剩余的机会打出最大的几个攻击 ...

随机推荐

  1. 9、Java ConcurrentModificationException异常原因和解决方法

    Java ConcurrentModificationException异常原因和解决方法 在前面一篇文章中提到,对Vector.ArrayList在迭代的时候如果同时对其进行修改就会抛出java.u ...

  2. 韩国KT软件NB-IOT开发记录V150(2)FOTA差分包生成

    1. 生成差分包

  3. Updating Homebrew... 长时间不动解决方法

    确保你已安装Homebrew 依次输入下面的命令(注意:不要管重置部分的命令,这里原作者贴出来.我也贴出来是以防需要重置的时候有参考操作命令) 替换brew.git: cd "$(brew ...

  4. 使用nmon监控得出网络实时速度以及最大、最小、平均网络传送速度

    首先我们得搞清楚几个概念,即什么是网速?什么是带宽? 举两个个例子: 1.家里装网线,宽带提供商说我们的带宽是100兆. 2.用迅雷下载电影,迅雷显示实时的下载速度是每秒3兆,或者说是3MB/s. 这 ...

  5. Chromium添加一段新字符串

    参考:https://groups.google.com/a/chromium.org/forum/#!searchin/chromium-dev/tclib%7Csort:relevance/chr ...

  6. hdu1061Rightmost Digit(快速幂取余)

    Rightmost Digit Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others)T ...

  7. Selenium WebDriver(Python)API

    1.通过示例介绍Selenium-WebDriver 一个简单的入门方法就是这个例子,它在Google上搜索术语“Cheese”,然后将结果页面的标题输出到控制台. java csharp pytho ...

  8. JVM常见配置

    堆设置 -Xms:初始堆大小 -Xmx:最大堆大小 -XX:NewSize=n:设置年轻代大小 -XX:NewRatio=n:设置年轻代和年老代的比值.如:为3,表示年轻代与年老代比值为1:3,年轻代 ...

  9. 基于Python的接口自动化

    第一步 Python的安装配置 打开官网: https://www.python.org/downloads/ 目前官网上已经更新到3.6.1啦,有两个版本,大家可以按自己喜欢的去下载,我自己选择的是 ...

  10. java基础-Comparator接口与Collections实现排序算法

    java 排序Comparable和Comparator使用 java提供了两个排序用的接口Comparable和Comparator,一般情况下使用区别如下: Comparable 接口用于类的固定 ...