IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> [Wannafly 28] msc的背包 (生成函数 组合数) -> 正文阅读

[数据结构与算法][Wannafly 28] msc的背包 (生成函数 组合数)

题意

n n n 种体积为 1 1 1 的物品和 m m m 种体积为 2 2 2 的物品,求选择物品的体积为 k k k 的方案数

998244353 998244353 998244353 取模

( 1 ≤ n , m ≤ 1 0 6 , 1 ≤ k ≤ 9 × 1 0 8 ) (1 \le n, m \le 10 ^ 6,1 \le k \le 9 \times 10 ^ 8) (1n,m106,1k9×108)

分析:

所有体积为 1 1 1 的生成函数为
F ( x ) = ( ∑ i = 0 ∞ x i ) n F(x) = \left ( \sum_{i = 0} ^ {\infty} x ^ i \right ) ^ n F(x)=(i=0?xi)n
所有体积为 2 2 2 的生成函数为
G ( x ) = ( ∑ i = 0 ∞ x 2 i ) m G(x) = \left ( \sum_{i = 0} ^ {\infty} x ^ {2i} \right ) ^ m G(x)=(i=0?x2i)m
那么组成的所有体积方案数为 F ( x ) × G ( x ) F(x) \times G(x) F(x)×G(x)

F ( x ) F(x) F(x) G ( x ) G(x) G(x) 写成形式幂级数的逆的形式就为
1 ( 1 ? x ) n ( 1 ? x 2 ) m \frac{1}{(1-x) ^ n(1 - x ^ 2) ^ m} (1?x)n(1?x2)m1?
为了使分母的形式一致,对分数上下乘 ( 1 + x ) n (1 + x) ^ n (1+x)n
( 1 + x ) n ( 1 ? x 2 ) n + m \frac{(1 + x) ^ n}{(1 - x ^ 2) ^ {n + m}} (1?x2)n+m(1+x)n?
再把 1 ( 1 ? x 2 ) n + m \dfrac{1}{(1 - x ^ 2) ^ {n + m}} (1?x2)n+m1? 转为一般形式 ∑ j = 0 ∞ ( j + n + m ? 1 n + m ? 1 ) x 2 j \sum\limits_{j = 0} ^ {\infty} \binom{j + n + m - 1}{n + m - 1} x ^ {2j} j=0?(n+m?1j+n+m?1?)x2j

那么 ( 1 + x ) n (1 + x) ^ n (1+x)n 也对应二项式展开 ∑ i = 0 n ( n i ) x i \sum\limits_{i = 0} ^ {n} \binom{n}{i} x ^ i i=0n?(in?)xi

因为我们要求第 k k k 项的系数,所以考虑 x i x ^ i xi x 2 j x ^ {2j} x2j 凑出 x k x ^ k xk 的所有项,

也就是 i + 2 j = k i + 2 j = k i+2j=k,整理出 j = k ? i 2 j = \dfrac{k - i}{2} j=2k?i?

由于 i ∈ [ 0 , n ] i \in [0, n] i[0,n] 所以可以 O ( n ) O(n) O(n) 枚举 i i i 的范围,即
∑ i = 0 n [ ( k ? i ) ? m o d ? 2 = 0 ] ( n i ) ( k ? i 2 + n + m ? 1 n + m ? 1 ) \sum_{i = 0} ^ {n} [(k - i) \bmod 2 = 0] \binom{n}{i} \binom{\dfrac{k - i}{2} + n + m - 1}{n + m - 1} i=0n?[(k?i)mod2=0](in?)(n+m?12k?i?+n+m?1?)
每次 k ? i 2 \dfrac{k - i}{2} 2k?i? 只会减少 1 1 1,所以对第二个的组合数可以递推求解,分母一定是 ( n + m ? 1 ) ! (n + m - 1)! (n+m?1)!,那么分子每次必定会减少 1 1 1,所以只需要维护 a a a 为第一次的 k ? i 2 + 1 \dfrac{k - i}{2} + 1 2k?i?+1,之后每次乘 a + n + m a + n + m a+n+m 的逆元再乘 a ? 1 a - 1 a?1 就是答案 ( a a a 每次自减 1 1 1)

代码:

#include <bits/stdc++.h>
#define int long long
using namespace std;
const int mod = 998244353;
int qmi(int a, int b) {
    int res = 1;
    while (b) {
        if (b & 1) res = res * a % mod;
        a = a * a % mod;
        b >>= 1;
    }
    return res;
}
vector<int> fact, infact;
void init(int n) {
    fact.resize(n + 1), infact.resize(n + 1);
    fact[0] = infact[0] = 1;
    for (int i = 1; i <= n; i ++) {
        fact[i] = fact[i - 1] * i % mod;
    }
    infact[n] = qmi(fact[n], mod - 2);
    for (int i = n; i; i --) {
        infact[i - 1] = infact[i] * i % mod;
    }
}
int C(int n, int m) {
    if (n < 0 || m < 0 || n < m) return 0;
    return fact[n] * infact[n - m] % mod * infact[m] % mod;
}
signed main() {
    cin.tie(0) -> sync_with_stdio(0);
    init(1e6);
    int n, m, k;
    cin >> n >> m >> k;
    int res = 0;
    int inv = 1;
    for (int i = 1; i <= n + m - 1; i ++) {
        inv = inv * i % mod;
    }
    inv = qmi(inv, mod - 2);
    int flag = 0, sum = 1, last = 0;
    for (int i = 0; i <= n; i ++) {
        if ((k - i) % 2 == 0) {
            if (!flag) {
                flag = 1;
                for (int j = 1; j <= n + m - 1; j ++) {
                    sum = sum * ((k - i) / 2 + j) % mod;
                }
                last = (k - i) / 2 + 1;
            } else {
                last --;
                sum = sum * qmi((k - i) / 2 + n + m, mod - 2) % mod;
                sum = sum * last % mod;
            }
            res = (res + C(n, i) * sum % mod * inv % mod) % mod;
        }
    }
    cout << res << endl;
}
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-05-06 11:13:30  更:2022-05-06 11:15:23 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/18 7:31:21-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码