0

0

c++中如何计算组合数C(n,m)_c++求排列组合方法汇总

尼克

尼克

发布时间:2026-01-04 13:07:02

|

348人浏览过

|

来源于php中文网

原创

应避免用std::tgamma或阶乘公式直接计算大数组合数,而采用迭代乘除法控制中间值;unsigned long long可安全计算至c(67,33),超限需用boost::multiprecision::cpp_int等高精度方案。

c++中如何计算组合数c(n,m)_c++求排列组合方法汇总

直接用 std::tgamma 计算大数 C(n, m) 容易精度丢失

n 超过 20 左右,用阶乘公式 C(n,m) = n! / (m! * (n-m)!) 直接算 doublelong double 会迅速溢出或失真。比如 std::tgamma(n+1)n=171 就返回 inf(超出 double 表示范围)。更糟的是,中间结果远大于最终结果,导致有效数字被截断。

实际应避免全阶乘:改用迭代乘除,边乘边除,控制中间值在合理范围内。

  • max(m, n-m) 开始向上累乘分子,向下累除分母,减少迭代次数
  • unsigned long long 可安全算到 C(67, 33) ≈ 1.4e19(接近 ULLONG_MAX
  • 超过这个范围必须上高精度(如 boost::multiprecision::cpp_int 或手写数组)

unsigned long long 迭代实现安全整数版 C(n,m)

这是最常用、零依赖、兼顾速度与可读性的方案。核心是把组合数拆成连乘积:C(n,m) = ∏_{i=1}^m (n - m + i) / i,并保证每一步除法整除(数学上成立)。

unsigned long long comb(unsigned n, unsigned m) {
    if (m > n) return 0;
    if (m > n - m) m = n - m; // 利用对称性减少循环
    unsigned long long res = 1;
    for (unsigned i = 0; i < m; ++i) {
        res = res * (n - i) / (i + 1); // 先乘后除,整除成立
    }
    return res;
}

注意:res * (n - i) 这一步仍可能溢出——所以务必在 n 较大时加溢出检查,或改用 __builtin_mul_overflow(GCC/Clang)。

立即学习C++免费学习笔记(深入)”;

需要支持 n > 67?用 boost::multiprecision::cpp_int

标准库不提供大整数,但 boost::multiprecision 是事实上的补充。它支持任意精度整数,且语法几乎和原生类型一致,适合科学计算或算法验证场景。

AI Undetect
AI Undetect

让AI无法察觉,让文字更人性化,为文字体验创造无限可能。

下载

需链接 -lboost_system(部分平台),头文件为 <boost></boost>

#include <boost/multiprecision/cpp_int.hpp>
using namespace boost::multiprecision;
cpp_int comb_big(unsigned n, unsigned m) {
    if (m > n) return 0;
    if (m > n - m) m = n - m;
    cpp_int res = 1;
    for (unsigned i = 0; i < m; ++i) {
        res *= (n - i);
        res /= (i + 1);
    }
    return res;
}

性能比 ULLONG_MAX 版慢 10–100 倍,但换来的是完全无溢出风险。调试时可直接 std::cout 打印完整十进制结果。

std::next_permutationstd::next_combination 不是标准库函数

很多人搜“C++ 排列组合库函数”,误以为有类似 std::next_combination 的东西。实际上只有 std::next_permutation(用于生成全排列),而组合没有对应标准函数。

若需枚举所有大小为 m 的子集(即所有组合),常见做法是:

  • std::vector<bool> mask(n, false)</bool>,设前 m 位为 true,再用 std::next_permutation(mask.begin(), mask.end()) 枚举所有掩码
  • 每次循环中,遍历 mask 收集下标,构造一个组合
  • 注意:该方法时间复杂度 O(C(n,m) × n),仅适用于 n ≤ 30m 不极端的情况

真正高频使用组合数的场景(比如 DP、数论题),几乎都只需求值而非枚举;枚举本身应按需定制,避免依赖不存在的“标准组合函数”。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
python如何计算数的阶乘
python如何计算数的阶乘

方法:1、使用循环;2、使用递归;3、使用math模块;4、使用reduce函数。更多详细python如何计算数的阶乘的内容,可以阅读下面的文章。

177

2023.11.13

python求阶乘教程大全
python求阶乘教程大全

本专题整合了python求阶乘相关教程,阅读专题下面的文章了解更多详细内容。

13

2025.11.08

python语言求阶乘
python语言求阶乘

本专题整合了python中阶乘相关教程,阅读专题下面的文章了解更多详细步骤。

43

2025.12.06

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

105

2025.10.23

页面置换算法
页面置换算法

页面置换算法是操作系统中用来决定在内存中哪些页面应该被换出以便为新的页面提供空间的算法。本专题为大家提供页面置换算法的相关文章,大家可以免费体验。

486

2023.08.14

Swift iOS架构设计与MVVM模式实战
Swift iOS架构设计与MVVM模式实战

本专题聚焦 Swift 在 iOS 应用架构设计中的实践,系统讲解 MVVM 模式的核心思想、数据绑定机制、模块拆分策略以及组件化开发方法。内容涵盖网络层封装、状态管理、依赖注入与性能优化技巧。通过完整项目案例,帮助开发者构建结构清晰、可维护性强的 iOS 应用架构体系。

2

2026.03.03

C++高性能网络编程与Reactor模型实践
C++高性能网络编程与Reactor模型实践

本专题围绕 C++ 在高性能网络服务开发中的应用展开,深入讲解 Socket 编程、多路复用机制、Reactor 模型设计原理以及线程池协作策略。内容涵盖 epoll 实现机制、内存管理优化、连接管理策略与高并发场景下的性能调优方法。通过构建高并发网络服务器实战案例,帮助开发者掌握 C++ 在底层系统与网络通信领域的核心技术。

5

2026.03.03

Golang 测试体系与代码质量保障:工程级可靠性建设
Golang 测试体系与代码质量保障:工程级可靠性建设

Go语言测试体系与代码质量保障聚焦于构建工程级可靠性系统。本专题深入解析Go的测试工具链(如go test)、单元测试、集成测试及端到端测试实践,结合代码覆盖率分析、静态代码扫描(如go vet)和动态分析工具,建立全链路质量监控机制。通过自动化测试框架、持续集成(CI)流水线配置及代码审查规范,实现测试用例管理、缺陷追踪与质量门禁控制,确保代码健壮性与可维护性,为高可靠性工程系统提供质量保障。

66

2026.02.28

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.6万人学习

C 教程
C 教程

共75课时 | 5.2万人学习

C++教程
C++教程

共115课时 | 20.4万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号