首页 > 其他分享 >[AGC021E] Ball Eat Chameleons 题解

[AGC021E] Ball Eat Chameleons 题解

时间:2024-02-07 20:33:41浏览次数:29  
标签:kMod cnt Ball return int 题解 变色龙 蓝球 AGC021E

Description

有 \(n\) 只变色龙,一开始都是蓝色。现在你喂了 \(k\) 次球,每次指定一只变色龙吃下你指定颜色的球。

一只变色龙从蓝色变成红色当且仅当它吃的红球比蓝球多;
一只变色龙从红色变成蓝色当且仅当它吃的蓝球比红球多。

求最后能使所有变色龙都变成红色的方案数。

两个方案不同当且仅当至少一次喂的球颜色不同(而不是喂的变色龙不同)。

注意:存在一次喂的变色龙不同的两个方案可能是相同的方案。

\(1\leq n,k\leq 5\times 10^5\)。

Solution

考虑对于一只变色龙,怎样给他喂球才能变成红色。

设喂了 \(a\) 个红球,\(b\) 个蓝球。

  1. \(a>b\):为红色。
  2. \(a<b\):为蓝色。
  3. \(a=b\):为不同于最后喂的颜色的另一个颜色,即最后的颜色异或 \(1\)。

首先如果 \(a-b\geq 2\),那么把多出来的 \(a-b-1\) 个球给别人一定不劣,所以这里只需要满足 \(a=b+1\) 或 \(a=b\) 且最后一次选了蓝球。

不妨设一个方案总共放了 \(R\) 个红球,\(B\) 个蓝球,则要满足条件一定要满足 \(R\geq B\)。

容易发现 \(a=b+1\) 的变色龙总共 \(R-B\) 个,\(a=b\) 的变色龙共 \(n-(R-B)\) 个,不妨设 \(cnt=n-(R-B)\)。

所以判断一个方案是否合法等价于能否在其中找到 \(cnt\) 个形如 RB 的匹配。

但是这个判定还不够简单。

注意到这些匹配一定是前 \(cnt\) 个 R 和后 \(cnt\) 个 B 按顺序进行匹配,那么对于倒数第 \(i\) 个 B,它前面一定有至少 \(cnt-i+1\) 个 R

所以对于每个前缀,他的 R 的个数 \(\geq\) B 的个数 \(-(B-cnt)\)。

那么把 R 看作 \(+1\),B 看作 \(-1\) 做一遍前缀和,只要满足每个前缀和 \(\geq R-n\) 即可。

这就转化为了一个类似卡特兰数的东西,答案就是:

\[\binom{R+B}{R}-\binom{R+B}{n+B-R-1} \]

这里对于 \(R=B\) 要特判,因为最后一步必须是 \(B\),所以只要把 \(B-1\) 再做上面那个式子即可。

时间复杂度:\(O(k)\)。

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;
}

标签:kMod,cnt,Ball,return,int,题解,变色龙,蓝球,AGC021E
From: https://www.cnblogs.com/Scarab/p/18011249

相关文章

  • AT_abc270_g [ABC270G] Sequence in mod P 题解
    题目传送门前置知识大步小步算法解法递推式为\(x_{n}=(ax_{n-1}+b)\bmodp\),发现可以统一消去\(\bmodp\),只在最后参与计算。以下过程省去模运算。当\(x_{0}=t\)时,则\(n=0\)即为所求。当\(a=0,x_{0}\net\)时,递推式转化为\(x_{n}=b\bmodp\)。若\(b=t\),则......
  • UVA10225 Discrete Logging 题解
    题目传送门前置知识大步小步算法题意多组询问,每次询问依次给定\(p,a,b\),求\(a^{x}\equivb\pmod{p}\)的最小非负整数解,其中\(a,p\)互质。解法BSGS板子题,不做过多介绍。貌似本题比P3846[TJOI2007]可爱的质数/【模板】BSGS和BZOJ3239DiscreteLogging数据较强......
  • [AGC041F] Histogram Rooks 题解
    题目链接点击打开链接题目解法好牛(难)的题!!!所有都被覆盖不难想到容斥暴力的容斥是钦定集合\(S\)中的位置都没被覆盖,然后把不能填棋子的点都去掉,假设剩下的不受限制的点的个数为\(c\),则答案为\(\sum2^c(-1)^{|S|}\)这个暴力是很难直接优化的如果一列被有点被钦定了,那么......
  • 【洛谷 P2670】[NOIP2015 普及组] 扫雷游戏 题解(模拟)
    [NOIP2015普及组]扫雷游戏题目背景NOIP2015普及组T2题目描述扫雷游戏是一款十分经典的单机小游戏。在行列的雷区中有一些格子含有地雷(称之为地雷格),其他格子不含地雷(称之为非地雷格)。玩家翻开一个非地雷格时,该格将会出现一个数字——提示周围格子中有多少个是地雷格。游戏的......
  • UVA12024 Hats 题解
    题目传送门前置知识错位排列题意有\(t\)组询问,每次询问给定一个\(n\),表示有\(n\)个人,每人各有一个属于自己的帽子,求所有人都带错帽子的概率(不要求约分至最简形式)。解法错排板子题,\(\dfrac{D_{n}}{A_{n}^{n}}\)即为所求。代码#include<bits/stdc++.h>usingnamespac......
  • CF231E 题解
    本文采用CCBY-NC-SA4.0协议发布。前言提供一个圆方树做法。孩子圆方树学傻了,忘了还有缩点这回事。正文建圆方树。考虑一条圆方树上的路径,哪些点对答案有贡献:方点,这表示路径经过一个环,方案数\(\times2\).旁边有方点的圆点。这表示走到这时可以选择在环上绕一圈,方......
  • 洛谷P10136 暨 USACOJan2024S T3 题解
    题意简述原题已经很简了,没有什么简述的必要了。思维路径请注意本题解可以保证正确性但不保证如果有极端的Hack数据能够通过。拿到这道题上来的暴力想必是很容易的,即枚举每个\(L\)判断是否合法。接着我们就考虑优化,减少需要枚举的\(L\)的量。题目中要求余数最多有\(3\)......
  • CF1408F Two Different 题解
    解题思路先考虑如何将一堆数变为相同的。显然,这里有一个条件\(n=2^k,k\in\mathbbZ\),证明如下:每次操作只能将两个数变为相同的,那么一个数在使得其他数变为相同数的操作中(我们不妨将所有数进行这种操作称为一轮操作),一个数最多被使用一次;按照错位操作,即第一轮\(1\)和\(2\)......
  • CF1428D Bouncing Boomerangs 题解
    解题思路很简单的贪心。观察发现以下性质:当\(a_i=2\)时,这一行一定只有两个目标,且第二个目标一定位于一个\(a_j=1\)的格子内;当\(a_i=3\),那么当前列右边某一列发生转向的地方,\(a_j\not=0\);那么这道题就基本已经做出来了。因为\(a_i=3\)的格子转向格可以在任意非\(0\)......
  • CF1401E Divide Square 题解
    解题思路其实多看几组也能发现块数等于交点的数量加上两个端点都在边上的线段数量再加一。证明如下(图见样例):对于两条只有一个端点位于边上的线段,因为保证有一个端点位于边上,那么这两条线段的交点一定会和已存在的点、边构成一个新的矩形;对于其中有一条为两个端点均位于边上的......