[AGC021E] Ball Eat Chameleons 题解
Description
有
一只变色龙从蓝色变成红色当且仅当它吃的红球比蓝球多;
一只变色龙从红色变成蓝色当且仅当它吃的蓝球比红球多。
求最后能使所有变色龙都变成红色的方案数。
两个方案不同当且仅当至少一次喂的球颜色不同(而不是喂的变色龙不同)。
注意:存在一次喂的变色龙不同的两个方案可能是相同的方案。
Solution
考虑对于一只变色龙,怎样给他喂球才能变成红色。
设喂了
:为红色。 :为蓝色。 :为不同于最后喂的颜色的另一个颜色,即最后的颜色异或 。
首先如果
不妨设一个方案总共放了
容易发现
所以判断一个方案是否合法等价于能否在其中找到 RB
的匹配。
但是这个判定还不够简单。
注意到这些匹配一定是前 R
和后 B
按顺序进行匹配,那么对于倒数第 B
,它前面一定有至少 R
。
所以对于每个前缀,他的 R
的个数 B
的个数
那么把 R
看作 B
看作
这就转化为了一个类似卡特兰数的东西,答案就是:
这里对于
时间复杂度:
Code
#include <bits/stdc++.h> // #define int int64_t const int kMaxN = 5e5 + 5, kMod = 998244353; int n, k; int inv[kMaxN], fac[kMaxN], ifac[kMaxN]; int add(int x, int y) { return (x + y) >= kMod ? (x + y - kMod) : (x + y); } int sub(int x, int y) { return (x < y) ? (x - y + kMod) : (x - y); } int C(int m, int n) { if (m < n || m < 0 || n < 0) return 0; return 1ll * fac[m] * ifac[n] % kMod * ifac[m - n] % kMod; } void prework() { fac[0] = ifac[0] = fac[1] = ifac[1] = inv[1] = 1; for (int i = 2; i <= k; ++i) { inv[i] = 1ll * (kMod - kMod / i) * inv[kMod % i] % kMod; fac[i] = 1ll * i * fac[i - 1] % kMod; ifac[i] = 1ll * inv[i] * ifac[i - 1] % kMod; } } void dickdreamer() { std::cin >> n >> k; if (k < n) return void(std::cout << "0\n"); prework(); int ans = 0; for (int i = (k + 1) / 2; i <= k; ++i) { int j = k - i; if (i == j) -- j; ans = add(ans, sub(C(i + j, i), C(i + j, n + j - i - 1))); } std::cout << ans << '\n'; } int32_t main() { #ifdef ORZXKR freopen("in.txt", "r", stdin); freopen("out.txt", "w", stdout); #endif std::ios::sync_with_stdio(0), std::cin.tie(0), std::cout.tie(0); int T = 1; // std::cin >> T; while (T--) dickdreamer(); // std::cerr << 1.0 * clock() / CLOCKS_PER_SEC << "s\n"; return 0; }
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步