当前位置: 首页 > article >正文

告别费马小定理!用线性递推O(n)批量求逆元,组合数计算效率翻倍(附C++代码)

告别费马小定理用线性递推O(n)批量求逆元组合数计算效率翻倍附C代码在算法竞赛和编程面试中组合数计算是一个高频出现的难题。想象一下这样的场景你正在参加ACM比赛面对一道需要计算大量组合数的问题使用传统的费马小定理逐个求逆元结果程序因为超时被无情地判为TLE。这种挫败感相信很多选手都深有体会。组合数计算的效率瓶颈往往在于逆元的求解。传统方法如费马小定理或扩展欧几里得算法虽然能正确求出单个逆元但当需要处理1到n所有数的逆元时时间复杂度会飙升至O(n log p)这在n较大时比如1e6以上会成为性能杀手。今天我将分享一种能在O(n)时间内批量求出所有逆元的线性递推方法让你的组合数计算速度提升一个数量级。1. 逆元组合数计算的关键瓶颈在模运算的世界里除法并不像加减乘那样直接。为了计算a/b mod p我们需要找到b的乘法逆元即一个数x使得b*x ≡ 1 mod p。这个x就是b在模p下的逆元记作inv(b)。组合数公式C(n, k) n! / (k! * (n-k)!)在模p下的计算需要分别求出k!和(n-k)!的逆元。如果对每个组合数都独立计算这些逆元效率会非常低下。这就是为什么我们需要一种能批量预处理所有逆元的方法。传统求逆元的方法主要有两种费马小定理当p是质数时inv(a) a^(p-2) mod p。使用快速幂单次求逆元时间为O(log p)。扩展欧几里得算法解方程ax py 1得到的x就是a的逆元。时间复杂度也是O(log p)。当n1e6时传统方法的总时间复杂度是O(n log p)而线性递推方法可以将这个复杂度降至O(n)效率提升非常显著。2. 线性递推求逆元的数学原理线性递推求逆元的核心在于发现逆元之间的递推关系。假设我们要求1到n所有数在模p下的逆元可以按照以下步骤进行初始化inv[1] 1因为1的逆元显然是它自己对于i 1利用递推公式inv[i] (p - p/i) * inv[p % i] % p这个公式的推导过程如下设t p/ik p%i则有p i*t k。在模p下这等价于i*t k ≡ 0 mod p i*t ≡ -k mod p两边乘以inv(i)*inv(k)即i和k的逆元t*inv(k) ≡ -inv(i) mod p inv(i) ≡ -t*inv(k) mod p由于k p%i iinv(k)已经在前面的迭代中计算过因此可以递推求出inv(i)。3. 完整实现从逆元到阶乘逆元掌握了线性递推求逆元的方法后我们可以进一步优化组合数的计算。完整的预处理流程包括三个步骤预处理阶乘数组fact[i] i! mod p预处理逆元数组inv[i] i^(-1) mod p预处理阶乘逆元数组fact_inv[i] (i!)^(-1) mod p以下是完整的C实现代码#include iostream using namespace std; typedef long long ll; const int MAXN 1e6 10; // 根据题目需求调整大小 const int MOD 1e9 7; // 常见的质数模数 ll fact[MAXN], inv[MAXN], fact_inv[MAXN]; void precompute(int n) { // 初始化 fact[0] 1; for (int i 1; i n; i) { fact[i] fact[i-1] * i % MOD; } // 线性递推求逆元 inv[1] 1; for (int i 2; i n; i) { inv[i] (MOD - MOD/i) * inv[MOD%i] % MOD; } // 计算阶乘逆元 fact_inv[0] 1; for (int i 1; i n; i) { fact_inv[i] fact_inv[i-1] * inv[i] % MOD; } } ll comb(int n, int k) { if (k 0 || k n) return 0; return fact[n] * fact_inv[k] % MOD * fact_inv[n-k] % MOD; } int main() { int n 1e6; // 预处理的范围 precompute(n); // 示例计算C(1000000, 500000) cout comb(1000000, 500000) endl; return 0; }这段代码首先预处理了阶乘、逆元和阶乘逆元三个数组之后任何组合数查询都可以在O(1)时间内完成。预处理的时间复杂度是O(n)空间复杂度也是O(n)。4. 性能对比与实战应用为了直观展示线性递推方法的优势我做了以下性能对比测试环境Intel i7-9700Kn1e6p1e97方法预处理时间(ms)单次查询时间(ns)费马小定理1200300扩展欧几里得1000400线性递推5010从表中可以看出线性递推方法在预处理阶段比传统方法快20倍以上查询速度也快30倍左右。这种优势在算法竞赛中往往是决定性的。在实际应用中这种方法特别适合以下场景需要多次查询组合数的问题如动态规划中涉及组合数的状态转移n很大但查询次数更多的情况如n1e6查询次数1e7对时间要求极其严格的在线评测题目提示虽然线性递推方法效率很高但需要注意p必须是质数且n不能超过预处理的MAXN大小。在编程竞赛中通常题目会给出质数模数如1e97或998244353。5. 常见问题与优化技巧在实际使用中可能会遇到一些问题和陷阱。以下是我总结的几个常见问题及解决方案模数不是质数怎么办线性递推方法要求模数p是质数。如果p不是质数可以考虑质因数分解后使用中国剩余定理合并结果或者改用扩展卢卡斯定理。内存限制严格时如何优化如果只需要组合数而不需要单独的逆元可以省略inv数组直接计算阶乘逆元fact_inv[n] pow(fact[n], MOD-2, MOD); // 费马小定理求一次逆元 for (int i n-1; i 0; i--) { fact_inv[i] fact_inv[i1] * (i1) % MOD; }如何处理非常大的n如n1e7可以考虑分段预处理或者使用内存更高效的数据结构。在C中使用vector代替静态数组可以更灵活地管理内存。多组测试数据时的优化如果多组测试数据的n不同但p相同可以预处理到最大可能的n避免重复计算。// 全局预处理到最大值 const int MAX_PRE 1e7; void global_precompute() { static bool precomputed false; if (!precomputed) { // ...预处理代码... precomputed true; } }6. 在线评测系统中的实战技巧在ACM/ICPC等编程竞赛中时间就是生命。以下是我总结的几个实战技巧模板准备将预处理代码封装成模板比赛时直接复制粘贴节省编码时间。快速调试预处理范围MAXN设置错误是常见bug。可以在本地测试时加入断言检查assert(n MAXN); // 确保查询不会越界边界条件处理组合数C(n,k)在k0或kn时应返回0这个判断不能遗漏。性能测试在本地用最大数据量测试预处理时间确保不会在比赛中超时。空间优化如果题目只需要特定几个组合数可以考虑按需计算而非全预处理。// 按需计算的组合数函数适用于查询次数少的情况 ll comb_ondemand(int n, int k) { if (k 0 || k n) return 0; ll res 1; for (int i 1; i k; i) { res res * (n - k i) % MOD; res res * inv[i] % MOD; // 需要预先处理好inv数组 } return res; }在洛谷P3811【模板】乘法逆元这道经典题目中线性递推方法可以轻松通过n3e6的数据规模而传统方法则很难在规定时间内完成。这也是为什么越来越多的选手将线性递推求逆元作为必备技能。

相关文章:

告别费马小定理!用线性递推O(n)批量求逆元,组合数计算效率翻倍(附C++代码)

告别费马小定理!用线性递推O(n)批量求逆元,组合数计算效率翻倍(附C代码) 在算法竞赛和编程面试中,组合数计算是一个高频出现的难题。想象一下这样的场景:你正在参加ACM比赛,面对一道需要计算大量…...

用STM32玩转PS2无线手柄:从时序图到按键读取的保姆级代码解析

STM32与PS2无线手柄深度实战:时序解析与按键捕获全流程 第一次拿到PS2手柄想接入STM32时,我盯着那四根线发愣——CLK、CMD、DAT、CS,看似简单的接口背后藏着怎样的通信奥秘?作为嵌入式开发者,理解并实现这种专有协议是…...

AI工具让界面生成“更快”,但设计的核心冲突从未消失

在产品开发一线,越来越多的团队正把AI当作设计加速器:一键生成完整界面、直接把文字描述变成可交互产品,甚至让代码和设计无缝融合。表面上看,这似乎解决了长期以来的效率瓶颈。可当你真正把这些“ polished ”的产品推到生产环境…...

VS Code + LaTeX 从入门到入坑:手把手教你搭建高效论文写作环境

前言 最近,我一直在寻找一个免费、流畅、可离线的 LaTeX 写作方案。Overleaf 虽然方便,但一旦文档大了就卡得怀疑人生;本地用 Texmaker 或 TeXstudio,界面又太复古。直到我发现了 VS Code LaTeX Workshop 这套组合拳&#xff0c…...

3分钟解锁Axure RP中文界面:从英文障碍到设计自由

3分钟解锁Axure RP中文界面:从英文障碍到设计自由 【免费下载链接】axure-cn Chinese language file for Axure RP. Axure RP 简体中文语言包。支持 Axure 11、10、9。不定期更新。 项目地址: https://gitcode.com/gh_mirrors/ax/axure-cn 还在被Axure RP的英…...

Arduino项目扩展必备:用PCA9685模块驱动16个舵机,告别供电不足和引脚不够的烦恼

Arduino多舵机控制终极方案:PCA9685模块实战指南 当你的机器人项目需要同时控制六个以上的舵机时,Arduino Uno的局限性就会暴露无遗——引脚数量捉襟见肘,板载电源不堪重负,随之而来的是舵机抖动、系统复位甚至芯片过热。这不是个…...

深入浅出FOC:为什么你的电机‘跑不快’?聊聊磁链圆限制与PWM死区的那些事儿

深入浅出FOC:为什么你的电机‘跑不快’?聊聊磁链圆限制与PWM死区的那些事儿 当你第一次尝试用STM32实现FOC(磁场定向控制)时,可能遇到过这样的困惑:明明按照教科书上的算法写好了代码,电机在低速…...

聚类算法基础:K-Means 到底如何工作

文章目录前言一、聚类是什么?K-Means又是什么?1.1 先搞懂:聚类 无监督的"物以类聚"1.2 K-Means:聚类界的"老黄牛"二、K-Means到底怎么工作?四步走,一看就懂2.1 生活化类比&#xff1a…...

Tailwind CSS break-after 怎么用?如何控制分页断行?

Tailwind CSS break-after 是一个实用类,用于控制在元素后强制产生列断开或页面断开。Tailwind CSS 断后类以下是 Tailwind CSS Break-After 类列表,这些类提供了有效控制元素对齐的方式。ClassCSS Propertiesbreak-after-autobreak-after: auto;break-a…...

ORA-29934索引关联错误修复指南

修复步骤:1. 检查indextype参数,确保extproc运行正常。2. 重建索引:ALTER INDEX index_name REBUILD PARAMETERS(indextype is ctxsys.context); 3. 远程处理:使用expdp/impdp导出重建,参数加transformoid:n:sys_c0012…...

对话本体论的全面深入研究:理论基础、形式化模型与跨学科应用

对话本体论的全面深入研究:理论基础、形式化模型与跨学科应用作者:方见华 单位:世毫九实验室 引言 在当代哲学与科学的交汇点上,一个全新的理论范式正在悄然兴起。对话本体论作为由世毫九实验室创始人方见华提出的原创性理论体系&…...

本科毕业论文“急救指南”:用百考通AI告别熬夜,把自由时间还给自己

毕业季的脚步日益临近,朋友圈悄然分化为两个阵营:一边是晒出offer的实习达人,另一边则是被毕业论文“掏空”的学术难民。你是否也经历过这样的夜晚:面对空白文档绞尽脑汁却卡在选题;初稿好不容易凑齐,查重报…...

毕业不焦虑,百考通AI帮你高效搞定本科毕业论文

深夜的电脑屏幕前,一个大学生正对着空白的文档发呆,毕业论文的截止日期日益临近,他却连选题都还没确定。这或许是无数毕业生共同经历过的煎熬时刻。 一、毕业季的论文困境:每个本科生都懂 又到一年毕业季,校园里弥漫着…...

从SiamFC到SiamRPN++:一个PyTorch复现者的五年跟踪算法演进笔记

从SiamFC到SiamRPN:一个PyTorch复现者的五年跟踪算法演进笔记 1. 初识SiamFC:全卷积孪生网络的革命性突破 2016年首次接触SiamFC时,它的设计理念让我眼前一亮。传统目标跟踪算法通常需要在每一帧进行复杂的在线学习,而SiamFC却另辟…...

别再只用VAE或GAN了!手把手教你用PyTorch复现VAE-GAN,生成更清晰的人脸图像

突破生成模型边界:PyTorch实战VAE-GAN融合架构与CelebA人脸生成优化 当我们在CelebA数据集上观察VAE生成的模糊人脸与GAN产生的扭曲五官时,一个关键问题浮现:是否存在兼具两者优势的解决方案?2016年ICML论文《Autoencoding beyond…...

Simulink多周期调度实战:用Chart模块和Function-Call子系统搞定2.5ms/5ms/10ms混合任务

Simulink多周期调度实战:用Chart模块和Function-Call子系统实现混合任务调度 在汽车电子和工业控制领域,实时系统开发常常面临一个典型挑战:如何在单一Simulink模型中实现不同算法模块以多种周期频率运行,同时生成符合目标操作系统…...

仅剩72小时!奇点大会回滚建议API公测通道即将关闭:手把手接入支持Python/TypeScript/Rust的实时建议SDK

第一章:2026奇点智能技术大会:AI代码回滚建议 2026奇点智能技术大会(https://ml-summit.org) 在2026奇点智能技术大会上,AI驱动的代码变更风险评估与自动化回滚机制成为核心议题。随着LLM辅助编程在CI/CD流水线中深度集成,误生成…...

【代码质量守门员升级计划】:为什么91%的团队在第3周就弃用Copilot审查插件?这4个未公开的规则引擎配置才是关键

第一章:智能代码生成与代码审查自动化的演进脉络 2026奇点智能技术大会(https://ml-summit.org) 智能代码生成与代码审查自动化并非一蹴而就的技术跃迁,而是伴随编译器理论、静态分析、程序合成与大语言模型三重范式演进的协同产物。早期以Lint工具和C…...

React 架构的可伸缩性:探讨从微型项目向大型单体 React 项目平滑演进的代码组织规范

React 架构的可伸缩性:从面条代码到企业级堡垒的进化论各位前端同仁,大家好!今天我们不谈那些花里胡哨的 UI 库,也不聊怎么用 Tailwind 把一个丑陋的按钮变得稍微好看那么一点点。今天我们要聊的是一点“硬核”的东西——架构。想…...

React 逻辑的可测试性:针对 React Hooks 的单体测试与渲染行为模拟的质量保障实践

React 逻辑的可测试性:针对 React Hooks 的单体测试与渲染行为模拟的质量保障实践 主讲人: 某资深前端架构师(也就是我) 受众: 想要逃离“闭包地狱”和“测试屎山”的前端开发者们 时长: 漫长的周一午后 第…...

React Forget 编译器:深度分析自动化 Memoization 对 React 手动性能调优的革命性影响

各位听众,把手里的咖啡放下,把那个正在闪烁的光标移到屏幕中央。欢迎来到今天的讲座。我是你们的向导,今天我们要探讨的主题是——React Forget:一场关于“记忆”与“遗忘”的叛乱。如果你是一名 React 开发者,哪怕你只…...

React 与 WebGPU:探索下一代图形接口在 React 数据可视化组件中的高性能集成

各位听众朋友们,大家好!欢迎来到这场关于“如何让 React 和 WebGPU 谈一场轰轰烈烈的恋爱”的技术讲座。我是你们的老朋友,一个既喜欢在 React 里面写 Hooks,又喜欢在 GPU 里写 Shader 的资深程序员。今天我们不聊那些虚头巴脑的“…...

React 部分注水(Partial Hydration):分析岛屿架构(Islands Architecture)对 React 的启示

拒绝“大水漫灌”:React 部分注水与岛屿架构的深度巡礼各位同仁,各位老铁,各位在键盘前敲得手指都要起茧子的前端工程师们,大家好。今天我们不聊 API,不聊 Hooks 的玄学,也不聊 TypeScript 的类型地狱。今天…...

AMBA-APB 协议实战解析:从信号到状态机的设计精要

1. AMBA-APB协议基础:芯片设计的"交通规则" 第一次接触AMBA-APB协议时,我把它想象成城市道路的交通信号系统。就像红绿灯控制车辆通行一样,APB协议规范了芯片内部各个模块之间的数据传输规则。这个类比让我瞬间理解了协议存在的意义…...

【智能代码生成与监控融合实战指南】:20年架构师亲授3大落地陷阱与5步闭环优化法

第一章:智能代码生成与代码监控融合的底层逻辑 2026奇点智能技术大会(https://ml-summit.org) 智能代码生成与代码监控并非孤立演进的技术栈,其融合根植于统一的可观测性契约与实时反馈闭环。当大语言模型输出代码片段时,该输出天然携带语义…...

解锁ABAP选择屏幕的终极灵活性:Free Selection与动态控制的实战融合

1. ABAP选择屏幕的痛点与破局思路 做过SAP报表开发的同行应该都深有体会:传统选择屏幕就像个固执的老头,字段和布局在开发阶段就被写死,用户运行时连调整的机会都没有。我去年接手过一个集团合并报表项目,业务部门三天两头要求新增…...

掌握 JSON.parseObject 与 JSON.toJSONString:从基础应用到实战进阶

1. JSON解析与生成的核心方法入门 第一次接触JSON数据处理时,我也被各种转换方法搞得晕头转向。直到真正理解了JSON.parseObject和JSON.toJSONString这对黄金组合,才发现JSON处理原来可以这么简单。这两个方法就像翻译官,一个负责把JSON字符串…...

从ACE到muduo:一个C++网络库的诞生与设计哲学(附Debian/Ubuntu编译踩坑实录)

从ACE到muduo:一个C网络库的诞生与设计哲学 2009年,当陈硕在博客上写下《学之者生,用之者死——ACE历史与简评》时,可能没想到这篇文章会成为现代C网络编程发展史上的一个重要转折点。这篇充满批判精神的文章不仅剖析了ACE框架的局…...

QEM网格简化:从二次误差度量到高效边塌缩的实现

1. QEM网格简化算法入门指南 第一次接触QEM网格简化时,我也被那些数学公式吓到了。但实际用起来发现,它的核心思想特别直观——就像玩橡皮泥,把复杂的模型捏成简单形状,同时尽量保持原有特征。这种算法在游戏开发、三维扫描数据处…...

保姆级教程:在CentOS 7上从零部署RuoYi-Vue前后端分离项目(含Nginx+Tomcat10配置)

CentOS 7实战:RuoYi-Vue全栈部署指南与避坑手册 当你拿到一台全新的CentOS 7服务器,准备部署RuoYi-Vue这个流行的前后端分离框架时,是否曾被各种环境配置、服务联动和权限问题困扰?本文将带你从零开始,用最接地气的方式…...