0

0

c++模板递归怎么用 c++编译期计算阶乘【实例】

尼克

尼克

发布时间:2025-12-25 16:37:02

|

645人浏览过

|

来源于php中文网

原创

c++kquote>C++模板递归实现编译期阶乘通过类模板特化与实例化在编译阶段计算,核心是通用模板定义递归逻辑、特化模板终止递归;支持变量模板(C++14)和constexpr函数(C++14起)等更现代写法,并需注意溢出、深度限制及输入校验。

c++模板递归怎么用 c++编译期计算阶乘【实例】

c++模板递归实现编译期阶乘

用C++模板递归计算阶乘,本质是利用类模板的特化和实例化过程,在编译阶段完成数值推导,不生成任何运行时代码。核心在于:定义通用模板(递归情况)+ 显式特化(递归终止)。

基础写法:类模板 + 静态常量成员

这是最经典、最易理解的方式。通过 enumconstexpr static 成员保存结果:

// C++11 及以上写法(推荐)

template
struct factorial {
    static constexpr unsigned long long value = N * factorial::value;
};

// 递归终止:0! = 1 和 1! = 1(可合并为一个特化) template<> struct factorial<0> { static constexpr unsigned long long value = 1; };

// 使用示例 static_assert(factorial<5>::value == 120, "5! should be 120"); int x = factorial<4>::value; // 编译期求得 24,x 是运行时整数

更现代写法:变量模板(C++14)

避免重复写 ::value,语法更简洁,语义更清晰:

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

Img.Upscaler
Img.Upscaler

免费的AI图片放大工具

下载
  • 需先定义类模板(含特化),再导出变量模板
  • 变量模板本身不能偏特化,所以仍依赖类模板特化逻辑
template
struct factorial_impl {
    static constexpr unsigned long long value = N * factorial_impl::value;
};

template<> struct factorial_impl<0> { static constexpr unsigned long long value = 1; };

// 变量模板:直接访问 template constexpr unsigned long long factorial = factorial_impl::value;

// 使用 static_assert(factorial<6> == 720, "");

注意事项与边界处理

模板递归不是万能的,需主动防范溢出和非法输入:

  • 负数不支持:无符号类型(unsigned int)天然拒绝负值,但若用 intN-1N==0 时会回绕成大正数,导致无限实例化失败(编译错误
  • 溢出静默发生factorialunsigned long long 下就溢出(实际值超 2^64),但编译器通常不报错,只得到错误结果;建议配合 static_assert 检查上限
  • 深度限制:过大的 N(如 >1000)可能触发编译器模板递归深度限制(如 GCC 默认 900),可用 -ftemplate-depth= 调整,但应优先优化逻辑

// 加入上限检查示例(C++17)

template
struct factorial {
    static_assert(N < 21, "factorial: N too large, risk of overflow");
    static constexpr unsigned long long value = N * factorial::value;
};
template<> struct factorial<0> { static constexpr unsigned long long value = 1; };

进阶:函数模板递归(C++14 constexpr 函数)

相比类模板,constexpr 函数写法更直观,且天然支持运行时调用(一鱼两吃):

constexpr unsigned long long factorial(unsigned int n) {
    return n <= 1 ? 1 : n * factorial(n - 1);
}

// 编译期使用 constexpr auto f5 = factorial(5); // OK,值为 120 static_assert(f5 == 120, "");

// 运行时也可用 int y = factorial(7); // 生成运行时调用(或内联)

注意:该函数在 C++14 中已允许循环和递归;C++11 仅支持单条 return 表达式,无法写递归版本。

相关专题

更多
java基础知识汇总
java基础知识汇总

java基础知识有Java的历史和特点、Java的开发环境、Java的基本数据类型、变量和常量、运算符和表达式、控制语句、数组和字符串等等知识点。想要知道更多关于java基础知识的朋友,请阅读本专题下面的的有关文章,欢迎大家来php中文网学习。

1428

2023.10.24

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

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

154

2023.11.13

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

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

8

2025.11.08

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

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

15

2025.12.06

string转int
string转int

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

311

2023.08.02

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

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

510

2024.08.29

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

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

46

2025.08.29

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

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

177

2025.08.29

苹果官网入口直接访问
苹果官网入口直接访问

苹果官网直接访问入口是https://www.apple.com/cn/,该页面具备0.8秒首屏渲染、HTTP/3与Brotli加速、WebP+AVIF双格式图片、免登录浏览全参数等特性。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

10

2025.12.24

热门下载

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

精品课程

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

共94课时 | 5.3万人学习

C 教程
C 教程

共75课时 | 3.7万人学习

C++教程
C++教程

共115课时 | 9.8万人学习

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

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