0

0

C++如何使用std::result_of推导函数调用返回类型?(C++17前替代invoke_result)

穿越時空

穿越時空

发布时间:2026-03-01 14:31:49

|

299人浏览过

|

来源于php中文网

原创

应使用std::result_of::type,必须对每个参数用std::declval,函数类型f需可调用且不能是重载名或未定义模板别名。

c++如何使用std::result_of推导函数调用返回类型?(c++17前替代invoke_result)

std::result_of 在 C++11/14 中怎么写才不报错?

它早被弃用了,但老项目里躲不开。关键不是“怎么用”,而是“怎么写才让编译器不吐 std::result_of<f>::type</f> 未定义”。

常见错误现象:error: 'type' in 'struct std::result_of<...>' does not name a type</...> —— 这几乎全是参数类型没加 std::declval 导致的。

  • 必须对每个参数用 std::declval<arg>()</arg>,不能直接写 Arg;比如 std::result_of<f double>::type</f> 是错的,得写成 std::result_of<f>(), std::declval<double>())>::type</double></f>
  • 函数类型 F 必须可调用:普通函数指针、成员函数指针、functor 类型都行,但不能是重载函数名(会歧义),也不能是未定义的模板别名
  • 如果 F 是成员函数指针,第一个参数得是对象(或指针)类型,例如 std::result_of<decltype int>::type</decltype> 要写成 std::result_of<decltype>(), std::declval<int>())>::type</int></decltype>

为什么 std::result_of 会被 std::invoke_result 替代?

根本原因:std::result_of 的 SFINAE 友好性差,且语义模糊 —— 它试图模拟“调用”,但实际依赖的是一个过时的可调用性检测规则(C++11 初版),遇到引用折叠、cv 限定符、返回引用等场景容易崩。

典型兼容性影响:

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

Booltool
Booltool

常用AI图片图像处理工具箱

下载
  • std::result_of<f></f> 在某些编译器(如旧版 GCC)里无法推导右值引用调用结果
  • noexcept 函数或返回 auto 的 lambda,std::result_of 常静默失败,而 std::invoke_result 明确支持
  • C++17 起,std::result_of 标准中被标记为 deprecated,Clang 10+ 和 GCC 9+ 默认开启 -Wdeprecated-declarations 警告

在 C++14 项目里安全迁移到 invoke_result 的最小改动

不是所有地方都能立刻升 C++17,但可以靠 std::invoke_result_t 的别名兼容层绕过去 —— 关键是别自己实现,用标准库已提供的兜底。

实操建议:

  • 头文件必须加 #include <type_traits></type_traits>(C++11 就有,但 std::invoke_result 需 C++17;不过你可以用条件编译 + 自定义别名)
  • 若编译器支持 C++17,直接用 std::invoke_result_t<f args...></f>,它不要求 std::declval,参数类型直写即可
  • 若只能用 C++14,定义兼容宏:
    #if __cplusplus >= 201703L
        using result_type = std::invoke_result_t<F, Args...>;
    #else
        using result_type = typename std::result_of<F(std::declval<Args>()...)::type;
    #endif

最容易被忽略的坑:std::result_of 对 void 返回值的处理

它能推导出 void,但一旦你拿这个 type 去做 sizeof 或模板特化,就可能触发未定义行为 —— 因为某些老标准库实现里,std::result_of<...>::type</...> 对 void 场景没做充分约束。

真实使用场景中,这会导致:

  • std::result_of 推导后接 std::enable_if_t<:is_same_v void>></:is_same_v>,结果 SFINAE 失效,编译器报硬错误而非静默丢弃
  • 在 trait 检测中嵌套使用(比如判断某函数是否返回 void),应优先改用 std::is_void_v<:invoke_result_t args...>></:invoke_result_t>,更稳
  • 如果必须用 std::result_of,记得加一层 std::decay_t 再判断:std::is_void_v<:decay_t std::result_of>::type>></:decay_t>

越老的代码越容易在这里卡住,因为没人想到 void 也会让类型推导变脆弱。

热门AI工具

更多
DeepSeek
DeepSeek

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

豆包大模型
豆包大模型

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

通义千问
通义千问

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

腾讯元宝
腾讯元宝

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

文心一言
文心一言

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

讯飞写作
讯飞写作

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

即梦AI
即梦AI

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

ChatGPT
ChatGPT

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

相关专题

更多
scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

411

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

357

2023.10.25

javascriptvoid(o)怎么解决
javascriptvoid(o)怎么解决

javascriptvoid(o)的解决办法:1、检查语法错误;2、确保正确的执行环境;3、检查其他代码的冲突;4、使用事件委托;5、使用其他绑定方式;6、检查外部资源等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

185

2023.11.23

java中void的含义
java中void的含义

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

125

2025.11.27

lambda表达式
lambda表达式

Lambda表达式是一种匿名函数的简洁表示方式,它可以在需要函数作为参数的地方使用,并提供了一种更简洁、更灵活的编码方式,其语法为“lambda 参数列表: 表达式”,参数列表是函数的参数,可以包含一个或多个参数,用逗号分隔,表达式是函数的执行体,用于定义函数的具体操作。本专题为大家提供lambda表达式相关的文章、下载、课程内容,供大家免费下载体验。

214

2023.09.15

python lambda函数
python lambda函数

本专题整合了python lambda函数用法详解,阅读专题下面的文章了解更多详细内容。

192

2025.11.08

Python lambda详解
Python lambda详解

本专题整合了Python lambda函数相关教程,阅读下面的文章了解更多详细内容。

60

2026.01.05

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

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

20

2026.02.28

Golang 工程化架构设计:可维护与可演进系统构建
Golang 工程化架构设计:可维护与可演进系统构建

Go语言工程化架构设计专注于构建高可维护性、可演进的企业级系统。本专题深入探讨Go项目的目录结构设计、模块划分、依赖管理等核心架构原则,涵盖微服务架构、领域驱动设计(DDD)在Go中的实践应用。通过实战案例解析接口抽象、错误处理、配置管理、日志监控等关键工程化技术,帮助开发者掌握构建稳定、可扩展Go应用的最佳实践方法。

15

2026.02.28

热门下载

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

精品课程

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

共94课时 | 10.4万人学习

C 教程
C 教程

共75课时 | 5.1万人学习

C++教程
C++教程

共115课时 | 19.8万人学习

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

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