热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

HDU6116路径计数优化

本文探讨了使用普通生成函数和指数生成函数解决组合与排列问题的方法,特别是在处理特定路径计数问题时的应用。文章通过详细分析和代码实现,展示了如何高效地计算在给定条件下不相邻相同元素的排列数量。

HDU 6116 路径计数优化


普通生成函数(OGF)主要用于解决组合问题,而指数生成函数(EGF)则更多应用于排列问题。这两种工具在解决特定类型的数学问题时非常有效,尤其是当涉及到计数问题时。


以一个问题为例,假设需要将 $a$ 个 $A$ 类型的对象分成多堆,每堆可以包含任意数量的对象,但不允许出现相邻两堆完全相同的情况。对于这样的分配,其方案数可以通过组合数 $C_{a-1}^{i-1}$ 来表示,其中 $i$ 是某一特定堆的数量。


进一步地,如果我们将每一堆视为一个独立的单元进行排列,则总的排列方式数量可以通过公式 $\frac{(i+j+k+l)!}{i!j!k!l!}$ 来计算,这里 $i, j, k, l$ 分别代表不同类型的堆的数量。值得注意的是,在这些排列中,即使所有的单个堆被视为不同的单位,仍然可能存在相邻的堆含有相同类型对象的情况,至少会有 $n-i-j-k-l$ 对相邻的相同堆。


为了准确计算不包含任何相邻相同堆的排列数,我们可以采用容斥原理。具体来说,通过枚举所有可能的 $i+j+k+l$ 的值,直接计算每个情况下的排列数,最终结果可以通过求和得到。数学表达式如下:


$ ans = \displaystyle \sum_{x=1}^{n} (-1)^{n-x} x! \sum_{i+j+k+l=x} \frac{C_{a-1}^{i-1} C_{b-1}^{j-1} C_{c-1}^{k-1} C_{d-1}^{l-1}}{i ! j ! k ! l !} $


上述公式实际上是四个指数生成函数的乘积形式,这为我们提供了一个有效的计算框架。


#include
using namespace std;
#define MAXN 200010
#define MOD 998244353
#define clr(a) memset(a, 0, sizeof a)
typedef long long ll;

// 快速幂算法
int quick_pow(int base, int exp) {
int result = 1;
while (exp) {
if (exp & 1) result = result * (ll)base % MOD;
base = base * (ll)base % MOD;
exp >>= 1;
}
return result;
}

// 计算组合数
int comb(int n, int m) {
if (m > n) return 0;
return (ll)J[n] * invJ[m] % MOD * invJ[n - m] % MOD;
}

int a, b, c, d;
int J[MAXN], invJ[MAXN], inv[MAXN];
int A[MAXN], B[MAXN], C[MAXN], D[MAXN];

int main() {
// 预处理阶乘和逆元
J[0] = inv[1] = invJ[0] = J[1] = invJ[1] = 1;
for (int i = 2; i inv[i] = (ll)(MOD - MOD / i) * inv[MOD % i] % MOD;
J[i] = (ll)J[i - 1] * i % MOD;
invJ[i] = (ll)invJ[i - 1] * inv[i] % MOD;
}

while (cin >> a >> b >> c >> d) {
clr(A), clr(B), clr(C), clr(D);
int n = a + b + c + d;
for (int i = 1; i <= a; ++i) A[i] = (ll)comb(a - 1, i - 1) * invJ[i] % MOD;
for (int i = 1; i <= b; ++i) B[i] = (ll)comb(b - 1, i - 1) * invJ[i] % MOD;
for (int i = 1; i <= c; ++i) C[i] = (ll)comb(c - 1, i - 1) * invJ[i] % MOD;
for (int i = 1; i <= d; ++i) D[i] = (ll)comb(d - 1, i - 1) * invJ[i] % MOD;

int len = 1, l = 0;
while (len <= n) len <<= 1, ++l;

// NTT预处理
int wn[2][MAXN];
for (int i = 1; i <(1 < int w0 = quick_pow(3, (MOD - 1) / (i <<1)), w1 = quick_pow(3, MOD - 1 - (MOD - 1) / (i <<1));
wn[0][i] = wn[1][i] = 1;
for (int j = 1; j wn[0][i + j] = (ll)wn[0][i + j - 1] * w0 % MOD,
wn[1][i + j] = (ll)wn[1][i + j - 1] * w1 % MOD;
}
int rev[MAXN];
for (int i = 1; i <(1 <> 1] >> 1) | ((i & 1) <
// 执行NTT变换
auto NTT = [&](int *A, int len, int flag) {
for (int i = 0; i for (int l = 1; l for (int i = 0; i for (int k = 0; k int t1 = A[i + k], t2 = (ll)A[i + l + k] * wn[flag][l + k] % MOD;
A[i + k] = (t1 + t2) % MOD;
A[i + l + k] = (t1 - t2 + MOD) % MOD;
}
if (flag == 1) {
int inv_len = quick_pow(len, MOD - 2);
for (int i = 0; i }
};

NTT(A, len, 0), NTT(B, len, 0), NTT(C, len, 0), NTT(D, len, 0);
for (int i = 0; i NTT(A, len, 1);

ll result = 0;
for (int i = 1; i <= n; ++i)
result += ((n - i & 1) ? -1ll : 1ll) * J[i] * A[i] % MOD,
result += MOD, result %= MOD;
cout < }
}

推荐阅读
  • 嵌入式开发环境搭建与文件传输指南
    本文详细介绍了如何为嵌入式应用开发搭建必要的软硬件环境,并提供了通过串口和网线两种方式将文件传输到开发板的具体步骤。适合Linux开发初学者参考。 ... [详细]
  • 探讨 HDU 1536 题目,即 S-Nim 游戏的博弈策略。通过 SG 函数分析游戏胜负的关键,并介绍如何编程实现解决方案。 ... [详细]
  • 深入解析动态代理模式:23种设计模式之三
    在设计模式中,动态代理模式是应用最为广泛的一种代理模式。它允许我们在运行时动态创建代理对象,并在调用方法时进行增强处理。本文将详细介绍动态代理的实现机制及其应用场景。 ... [详细]
  • Python 内存管理机制详解
    本文深入探讨了Python的内存管理机制,涵盖了垃圾回收、引用计数和内存池机制。通过具体示例和专业解释,帮助读者理解Python如何高效地管理和释放内存资源。 ... [详细]
  • 利用Selenium与ChromeDriver实现豆瓣网页全屏截图
    本文介绍了一种使用Selenium和ChromeDriver结合Python代码,轻松实现对豆瓣网站进行完整页面截图的方法。该方法不仅简单易行,而且解决了新版Selenium不再支持PhantomJS的问题。 ... [详细]
  • 解决TensorFlow CPU版本安装中的依赖问题
    本文记录了在安装CPU版本的TensorFlow过程中遇到的依赖问题及解决方案,特别是numpy版本不匹配和动态链接库(DLL)错误。通过详细的步骤说明和专业建议,帮助读者顺利安装并使用TensorFlow。 ... [详细]
  • 鼠标悬停出现提示信息怎么做
    概述–提示:指启示,提起注意或给予提醒和解释。在excel中会经常用到给某个格子增加提醒信息,比如金额提示输入数值或最大长度值等等。设置方式也有多种,简单的,仅为单元格插入批注就可 ... [详细]
  • Python自动化测试入门:Selenium环境搭建
    本文详细介绍如何在Python环境中安装和配置Selenium,包括开发工具PyCharm的安装、Python环境的设置以及Selenium包的安装方法。此外,还提供了编写和运行第一个自动化测试脚本的步骤。 ... [详细]
  • 本文介绍了如何通过Java代码计算一个整数的位数,并展示了多个基础编程示例,包括求和、平均分计算、条件判断等。 ... [详细]
  • 本题要求在一组数中反复取出两个数相加,并将结果放回数组中,最终求出最小的总加法代价。这是一个经典的哈夫曼编码问题,利用贪心算法可以有效地解决。 ... [详细]
  • 本篇文章介绍如何将两个分别表示整数的链表进行相加,并生成一个新的链表。每个链表节点包含0到9的数值,如9-3-7和6-3相加得到1-0-0-0。通过反向处理链表、逐位相加并处理进位,最终再将结果链表反向,即可完成计算。 ... [详细]
  • CentOS 系统管理基础
    本文介绍了如何在 CentOS 中查询系统版本、内核版本、位数以及磁盘分区的相关知识。通过这些命令,用户可以快速了解系统的配置和磁盘结构。 ... [详细]
  • 本文详细探讨了 PHP 中 method_exists() 和 is_callable() 函数的区别,帮助开发者更好地理解和使用这两个函数。文章不仅解释了它们的功能差异,还提供了代码示例和应用场景的分析。 ... [详细]
  • 本文探讨了C++编程中理解代码执行期间复杂度的挑战,特别是编译器在程序运行时生成额外指令以确保对象构造、内存管理、类型转换及临时对象创建的安全性。 ... [详细]
  • 本文详细介绍了如何解决 Microsoft SQL Server 中用户 'sa' 登录失败的问题。错误代码为 18470,提示该帐户已被禁用。我们将通过 Windows 身份验证方式登录,并启用 'sa' 帐户以恢复其访问权限。 ... [详细]
author-avatar
手机用户2502891053
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有