LOJ

BZOJ

洛谷

哪张能力牌能乘攻击啊,太nb了叭


显然如果有能力牌,那么应该选最大的尽可能的打出\(k-1\)张。

然后下面说的期望都是乘总方案数后的,即所有情况的和。然后\(w_i\)统一用\(A_i\)表示了。

\(Solution1\)

所以考虑枚举最终抽到了几张能力牌。那么我们要算:\(F(n,m)\)表示抽到\(n\)张攻击牌,打出最大的\(m\)张的期望伤害;\(G(n,m)\)表示抽到\(n\)张能力牌,打出最大的\(m\)张的期望倍数。

考虑怎么算\(F(n,m)\)。不妨枚举攻击最小的那张是什么。也就是把攻击牌从大到小排序,\(f[i][j]\)表示在前\(i\)张中选\(j\)张的期望伤害,其中第\(i\)张一定取。转移就再枚举一下次小的,\(f[i][j]=C_{i-1}^{j-1}A_i+\sum\limits_{k=j-1}^{i-1}f[k][j-1]\)。

\(G(n,m)\)同理。令\(g[i][j]\)表示从前\(i\)大的能力牌中选\(j\)张的期望倍数,其中第\(i\)张一定取。\(g[i][j]=A_i\sum\limits_{k=j-1}^{i-1}g[k][j-1]\)。

两个DP都可以前缀和优化。所以复杂度是\(O(n^2)\)的。

那么\(F,G\)的转移同样枚举选出来的最小的那张,\(F(n,m)=\sum\limits_{i=m}^{N}C_{N-i}^{n-m}f[i][m]\),\(G(n,m)=\sum\limits_{i=m}^{N}C_{N-i}^{n-m}g[i][m]\)。对于单个\(F/G(n,m)\)的计算是\(O(n)\)的。

然后\(Ans=\sum\limits_{i=1}^{k-1}G(i,i)F(m-i,k-i)+\sum\limits_{i=k}^NG(i,k-1)F(m-i,1)\)。我们只会用到\(O(n)\)个\(F/G\)的值,现算一下即可,总复杂度是\(O(n^2)\)的。

为了方便应该把\(f/g\)的一二维交换下=-=。

\(Solution2\)

设\(f[i][j]\)表示考虑了前\(i\)张牌,抽到了\(j\)张攻击牌的期望伤害。肯定是选出最大的若干张打出。将牌从小到大排序,然后根据\(j\)确定这张牌选不选,有(1年后发现Latex begin{cases}怎么不能用了?凑合看叭...)$$f[i][j]=f[i-1][j]+C_{i-1}^{j-1}A_i+\begin{cases}0,&m-j\geq k-1\ f[i-1][j-1],&m-j<k-1\end{cases}$$

考虑\(i\)的贡献,\(m-j\geq k-1\)时,只能取一张攻击牌,那取\(i\)就能保证打出的是最大的那张;否则取完\(i\)之后还可以在取\(j-1\)张的情况。然后再统计上以前的只考虑\(i-1\)张牌时的\(f[i-1][j]\)(注意这个=-=)。

\(g[i][j]\)表示考虑了前\(i\)张牌,抽到了\(j\)张能力牌的期望倍数。因为是选最大的\(\min(k-1,\ j)\)张打出,所以把能力牌从大到小排序,有$$g[i][j]=g[i-1][j]+\begin{cases}g[i-1][j-1]*A_i,&j<k\ g[i-1][j-1],&j\geq k\end{cases}$$

就是\(j<k\)时用\(A_i\),否则抽到\(A_i\)也不用。

那么答案就是\(\sum_{i=0}^mf[n][i]g[n][m-i]\)啦。

复杂度也是\(O(n^2)\),不过常数要更小。


但是懒得再写代码了,代码是\(Sol1\)的。不然还是有把握搞个BZOJRank1.2的。

//27376kb	9684ms
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
#include <functional>
#define mod 998244353
#define Mod(x) x>=mod&&(x-=mod)
#define Add(x,v) (x+=v)>=mod&&(x-=mod)
#define gc() getchar()
typedef long long LL;
const int N=1505; int A[N],B[N],f[N][N],g[N][N],C[N][N]; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-48,c=gc());
return now;
}
int F(int n,int m,int N)
{
LL ans=0; int *f=::f[m];
for(int i=m; i<=N; ++i) ans+=1ll*C[N-i][n-m]*f[i]%mod;
return ans%mod;
}
int G(int n,int m,int N)
{
LL ans=0; int *g=::g[m];
for(int i=m; i<=N; ++i) ans+=1ll*C[N-i][n-m]*g[i]%mod;
return ans%mod;
} int main()
{
C[0][0]=1;
for(int Ts=read(),mx=0; Ts--; )
{
int n=read(),m=read(),K=read();
if(mx<n)
{
for(int i=mx+1; i<=n; ++i)
{
C[i][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(C[i][j]);
}
mx=n;
}
for(int i=1; i<=n; ++i) A[i]=read();
for(int i=1; i<=n; ++i) B[i]=read();
std::sort(A+1,A+1+n,std::greater<int>()), std::sort(B+1,B+1+n,std::greater<int>()); g[0][0]=1;//!! K=1时会用到啊=-=
for(int i=1; i<=n; ++i) f[1][i]=B[i], g[1][i]=A[i];//or not
for(int i=2,lim=std::min(n,m); i<=lim; ++i)
{
int sf=0,sg=0;
for(int j=i; j<=n; ++j)
Add(sf,f[i-1][j-1]), f[i][j]=(1ll*C[j-1][i-1]*B[j]+sf)%mod,
Add(sg,g[i-1][j-1]), g[i][j]=1ll*A[j]*sg%mod;
}
LL ans=0;
for(int i=std::max(0,m-n); i<=n&&i<=m; ++i)//m-i<=n 枚举的是能力牌 可以是0啊=v=
if(i<K) ans+=1ll*G(i,i,n)*F(m-i,K-i,n)%mod;
else ans+=1ll*G(i,K-1,n)*F(m-i,1,n)%mod;
printf("%lld\n",ans%mod);
} return 0;
}

BZOJ.5467.[PKUWC2018]Slay the Spire(DP)的更多相关文章

  1. LOJ2538 PKUWC2018 Slay the Spire DP

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

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

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

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

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

  4. [LOJ2538] [PKUWC2018] Slay the Spire

    题目链接 LOJ:https://loj.ac/problem/2538 Solution 计数好题. 首先可以发现这题和期望没关系. 其次对于手上的一套牌,设我们有\(a\)张强化牌,那么: 如果\ ...

  5. [PKUWC2018] Slay the spire

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

  6. 题解-PKUWC2018 Slay the Spire

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

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

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

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

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

  9. BZOJ 5467 Slay the Spire

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

随机推荐

  1. uCosII中的任务

    任务基本概念 任务是一个接受操作系统管理的独立运行单元,在uCosII中类似与普通平台上的main()函数,需要自己来保护其因调用或中断二产生的断点,所以需要一个自己的私有堆栈,即任务堆栈: 任务有两 ...

  2. 【bzoj 2159】Crash 的文明世界

    Description Crash小朋友最近迷上了一款游戏——文明5(Civilization V).在这个游戏中,玩家可以建立和发展自己的国家,通过外交和别的国家交流,或是通过战争征服别的国家.现在 ...

  3. Angular 动画

    1.先做一个简单的例子  =>  定义一个div 从open渐变成closed ts:定义一个触发器 openClose,有两个状态 open 和 closed,均有对应的样式,再定义装换函数 ...

  4. LeetCode第二十三题-合并n个有序链表

    Merge k Sorted Lists 问题简介:合并k个已排序的链表并将其作为一个排序链表返回. 举例: 输入: [ 1->4->5, 1->3->4, 2->6 ] ...

  5. BUGKU login3

    先看的wp,呢么来复现一遍,emmmmmm,尝试一波,用户名输入admin后,密码随便输,发现提示password error,呢么填其他用户名的话,发现提示username does not exi ...

  6. 支持向量机SVM——专治线性不可分

    SVM原理 线性可分与线性不可分 线性可分 线性不可分-------[无论用哪条直线都无法将女生情绪正确分类] SVM的核函数可以帮助我们: 假设‘开心’是轻飘飘的,“不开心”是沉重的 将三维视图还原 ...

  7. CVE-2018-19386:SolarWinds数据库性能分析器中反射的XSS

    漏洞 在SolarWinds的11.1.457版中,"idcStateError.iwc"错误页面中存在Reflected Cross-Site Scripting漏洞,已经在版本 ...

  8. Python爬虫基础之正则表达式

    一.Python正则表达式的基本使用 Python 3 使用re模块可以实现大部分的正则表达式情况. 1.re.compile(pattern, flags=0) re.compile构建匹配规则并返 ...

  9. Python———pandas数据处理

    pandas模块 更高级的数据分析工具基于NumPy构建包含Series和DataFrame两种数据结构,以及相应方法 调用方法:from pandas import  Series, DataFra ...

  10. JDBC编程六部曲

    今天初学jdbc,明白了大致的编程流程,在此总结一下: JDBC编程可以分为六步——六部曲: * 第一步:注册驱动. * 1.1 获取驱动对象 * 1.2 注册驱动 * 第二步:获取数据库连接 * 第 ...