0

0

c++中如何实现组合数计算_c++计算排列组合C(n,k)的方法【实例】

穿越時空

穿越時空

发布时间:2026-01-15 16:24:04

|

369人浏览过

|

来源于php中文网

原创

推荐用迭代乘除法计算组合数:先取min(k, n−k),再边乘边除,用unsigned long long避免溢出和浮点误差,边界判断确保k∈[0,n],n>170时std::tgamma会溢出,不适用。

c++中如何实现组合数计算_c++计算排列组合c(n,k)的方法【实例】

直接用 std::tgamma 计算组合数容易出错

很多人第一反应是套公式 C(n,k) = n! / (k! × (n−k)!),再用 std::tgamma(n+1) 算阶乘。但这是危险的:浮点误差在 n > 17 时就明显,n > 170 会溢出为 inf,且 tgamma 对非整数输入无定义。实际项目中不推荐。

用迭代乘除法避免大数和溢出

核心思路是把 C(n,k) 展开为 k 项连乘再连除:C(n,k) = (n × (n−1) × … × (n−k+1)) / (1 × 2 × … × k),边乘边除,控制中间值始终在 long long 范围内(只要最终结果 ≤ 2⁶³−1)。

  • 先取 min(k, n−k) 替换 k —— 利用 C(n,k) = C(n,n−k) 减少循环次数
  • unsigned long long 存中间结果,更安全
  • 每步都做整除,确保不累积浮点误差
unsigned long long combination(int n, int k) {
    if (k < 0 || k > n) return 0;
    if (k == 0 || k == n) return 1;
    k = std::min(k, n - k);
    unsigned long long res = 1;
    for (int i = 1; i <= k; ++i) {
        res = res * (n - k + i) / i;
    }
    return res;
}

需要支持大 n?考虑模意义下的组合数

当 n 达到 1e5 或更大,且题目要求结果对某个质数(如 1e9+7)取模时,就不能用上面的迭代法——除法要转成乘逆元。此时预处理阶乘及其逆元是标准做法:

  • 预计算 fac[i] = i! mod MOD,O(n)
  • 用快速幂或线性递推求 inv_fac[n],再倒推所有 inv_fac[i]
  • C(n,k) ≡ fac[n] * inv_fac[k] % MOD * inv_fac[n-k] % MOD

注意:该方法仅适用于 MOD 是质数、且 n

BlackBox AI
BlackBox AI

AI编程助手,智能对话问答助手

下载

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

边界和类型问题比算法本身更容易翻车

写完函数别急着交,这几个点常被忽略:

  • nkint,但中间计算可能溢出 —— 必须用 unsigned long long 接收乘积
  • 输入可能为负或 k > n,返回 0 比崩溃更合理
  • 某些 OJ 编译器(如 MinGW)下 std::minintunsigned 混用会报警,显式转成同类型
  • 如果真要算 n=1000 以上的精确值,long long 不够,得上 boost::multiprecision::cpp_int 或手写高精度

组合数看着简单,但溢出、类型转换、边界判断这三块,占了实际调试时间的八成以上。

相关专题

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

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

166

2023.11.13

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

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

9

2025.11.08

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

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

27

2025.12.06

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

315

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

537

2024.08.29

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

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

52

2025.08.29

C++中int的含义
C++中int的含义

本专题整合了C++中int相关内容,阅读专题下面的文章了解更多详细内容。

197

2025.08.29

C++类型转换方式
C++类型转换方式

本专题整合了C++类型转换相关内容,想了解更多相关内容,请阅读专题下面的文章。

295

2025.07.15

Golang gRPC 服务开发与Protobuf实战
Golang gRPC 服务开发与Protobuf实战

本专题系统讲解 Golang 在 gRPC 服务开发中的完整实践,涵盖 Protobuf 定义与代码生成、gRPC 服务端与客户端实现、流式 RPC(Unary/Server/Client/Bidirectional)、错误处理、拦截器、中间件以及与 HTTP/REST 的对接方案。通过实际案例,帮助学习者掌握 使用 Go 构建高性能、强类型、可扩展的 RPC 服务体系,适用于微服务与内部系统通信场景。

8

2026.01.15

热门下载

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

精品课程

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

共94课时 | 6.8万人学习

C 教程
C 教程

共75课时 | 4万人学习

C++教程
C++教程

共115课时 | 12.3万人学习

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

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