0

0

c++20的std::remove_cvref相比std::decay有什么不同? (元编程)

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-01-12 14:01:20

|

186人浏览过

|

来源于php中文网

原创

std::remove_cvref用于精准剥离const、volatile和引用,还原表达式的“裸值类型”,避免std::decay意外退化数组或函数类型。

c++20的std::remove_cvref相比std::decay有什么不同? (元编程)

std::remove_cvref 是什么,它解决什么问题?

它只做三件事:去掉 constvolatile 修饰,再去掉引用(&&&)。不碰数组、函数类型、指针,也不做任何“退化”转换。它的目标非常明确:把一个任意表达式的类型还原成“裸值类型”,常用于完美转发后提取实参本体。

常见错误现象:用 std::decay 处理引用参数时,意外把数组转成指针、把函数转成函数指针,甚至把 int[5] 变成 int* —— 这在类型擦除或 trait 判断里会出错。

  • std::remove_cvref_tint
  • std::remove_cvref_tstd::string
  • std::remove_cvref_tconst int[3](数组类型保留)

std::decay 的行为远不止“去 cvref”

std::decay 模仿了函数形参类型的自动转换规则:除了去掉 cv 和引用,还会把数组转指针、函数类型转函数指针、并应用 std::remove_referencestd::remove_cv。它本质是为 std::functionstd::thread 等需要“值语义”的场景服务的。

使用场景:当你需要把任意实参(包括数组字面量、lambda、临时对象)统一转为可拷贝/可存储的类型时,std::decay 才是合适的。

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

  • std::decay_tint*
  • std::decay_tvoid(*)()
  • std::decay_tint(和 remove_cvref 结果一样)

什么时候该选 remove_cvref,什么时候必须用 decay?

关键看你在写什么:如果是在实现一个通用模板函数,想精准获取调用者传入的“原始值类型”(比如写自己的 forward_like 或判断某个成员是否可访问),就用 std::remove_cvref;如果是在封装可调用对象、做类型擦除(如 std::any 初始化、std::variant 构造),那 std::decay 是标准做法。

百度作家平台
百度作家平台

百度小说旗下一站式AI创作与投稿平台。

下载

性能与兼容性影响:两者都是编译期计算,无运行时开销。但误用 std::decay 可能导致类型信息丢失(比如数组长度),而误用 std::remove_cvref 则可能让函数类型无法被存储 —— 它根本不会帮你转成函数指针。

  • 要保留数组维度?→ 必须用 std::remove_cvref
  • 要传给 std::thread 构造函数?→ 必须用 std::decay
  • std::string_view 参数做类型推导?→ 通常用 std::remove_cvref,避免把 const char* 错误退化

一个典型误用对比示例

下面这个模板本意是提取参数“去掉引用和 cv 后的类型”,但用了 decay 就悄悄改变了语义:

template
struct wrapper {
    using type = std::decay_t; // ❌ 数组变指针,函数变指针
    // 应该是:using type = std::remove_cvref_t; ✅
};

比如 wrapper::type 在用 decay 时是 int*,用 remove_cvref 时才是 int[10]。这种差异在 SFINAE 或 concept 约束里会直接导致匹配失败。

最容易被忽略的是:std::remove_cvref 不处理指针——int* const& 经它处理后是 int*,而 std::decay 对它结果相同;但一旦涉及数组或函数,区别立刻暴露。别凭直觉猜,遇到不确定的类型,用 static_assert(std::is_same_v<...>) 实测。

相关专题

更多
c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

520

2023.09.20

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是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

534

2024.08.29

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

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

51

2025.08.29

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

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

194

2025.08.29

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

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

175

2023.11.23

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

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

96

2025.11.27

c++中volatile关键字的作用
c++中volatile关键字的作用

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

67

2025.10.23

Java 项目构建与依赖管理(Maven / Gradle)
Java 项目构建与依赖管理(Maven / Gradle)

本专题系统讲解 Java 项目构建与依赖管理的完整体系,重点覆盖 Maven 与 Gradle 的核心概念、项目生命周期、依赖冲突解决、多模块项目管理、构建加速与版本发布规范。通过真实项目结构示例,帮助学习者掌握 从零搭建、维护到发布 Java 工程的标准化流程,提升在实际团队开发中的工程能力与协作效率。

6

2026.01.12

热门下载

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

精品课程

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

共578课时 | 45.1万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 1.0万人学习

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

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