LOJ2538. 「PKUWC2018」Slay the Spire【组合数学】

LINKc++


思路

首先由于式子后面把方案数乘上了spa

因此其实只用输出全部方案的攻击力总和code

而后很显然能够用强化牌就尽可能用排序

由于每次强化至少把下面的牌翻一倍,确定是更优的get

而后就只有两种状况input

  • 强化牌数量少于k
  • 强化牌数量大于等于k

根据乘法原理,设\(f_{i,j}\)是选i张强化牌用j张的倍数总和,\(g_{i,j}\)是选i张攻击用j张的倍数总和it

\(ans+=f_{k,k}*g_{m-i,m-k}\)class

\(ans+=f_{i,k-1}*g_{m-i,1}\)原理

而后f的计算能够量化大小这个东西,就是先排序sort

dp出选了i个数,最后一个在j的方案数,这样前面的j各类不可能选出其余数,对于后面的数直接组合数计算就能够了


#include<bits/stdc++.h>

using namespace std;

const int Mod = 998244353;

const int N = 3e3 + 10;

int n, m, k, a[N], b[N], c[N][N];
int sum[N], f[N][N], g[N][N];

int add(int a, int b) {
  return (a += b) >= Mod ? a - Mod : a;
} 

int mul(int a, int b) {
  return 1ll * a * b % Mod;
} 

void init() {
  for (int i = 0; i < N; i++) c[i][0] = 1;
  for (int i = 1; i < N; i++) {
    for (int j = 1; j <= i; j++) {
      c[i][j] = add(c[i - 1][j], c[i - 1][j - 1]);
    }
  }
}

int calcf(int a, int b) { // 取a张用b张 
  if (a < b) return 0;
  if (!b) return c[n][a]; //**
  int res = 0;
  for (int i = 1; i <= n; i++)
    res = add(res, mul(f[b][i], c[n - i][a - b]));
  return res;
}

int calcg(int a, int b) {
  if (a < b) return 0;
  if (!b) return 0; //**
  int res = 0;
  for (int i = 1; i <= n; i++) 
    res = add(res, mul(g[b][i], c[n - i][a - b]));
  return res;
}

void solve() {
  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 + n + 1, [&](const int a, const int b) {return a > b;});
  sort(b + 1, b + n + 1, [&](const int a, const int b) {return a > b;});
  for (int i = 1; i <= n; i++) {
    f[1][i] = a[i];
    sum[i] = add(sum[i - 1], a[i]);
  }
  for (int i = 2; i <= n; i++) {
    for (int j = i; j <= n; j++)
      f[i][j] = mul(sum[j - 1], a[j]);
    for (int j = 1; j <= n; j++)
      sum[j] = add(sum[j - 1], f[i][j]);
  }
  for (int i = 1; i <= n; i++) {
    g[1][i] = b[i];
    sum[i] = add(sum[i - 1], b[i]);
  }
  for (int i = 2; i <= n; i++) {
    for (int j = i; j <= n; j++) {
      g[i][j] = add(mul(b[j], c[j - 1][i - 1]), sum[j - 1]);
    }
    for (int j = 1; j <= n; j++)
      sum[j] = add(sum[j - 1], g[i][j]);
  }
  int ans = 0;
  for (int i = max(0, m - n); i <= min(n, m); i++) {
    if (i < k) ans = add(ans, mul(calcf(i, i), calcg(m - i, k - i)));
    else ans = add(ans, mul(calcf(i, k - 1), calcg(m - i, 1)));
  }
  printf("%d\n", ans);
}

int main() {
#ifdef dream_maker
  freopen("input.txt", "r", stdin);
#endif
  init();
  int T; scanf("%d", &T);
  while (T--) solve();
  return 0;
}
相关文章
相关标签/搜索