C++完美转发

完美转发

什么是完美转发 (Perfect Forwarding)?std::forwardstd::move 有何本质区别?

问题背景:为什么需要完美转发?

假设编写一个"包装函数" (wrapper),它的功能是接收任何参数,然后将其原封不动地传递给另一个"目标函数" (target)

c++ 复制代码
void target(int& x) { std::cout << "lvalue reference called\n"; }
void target(int&& x) { std::cout << "rvalue reference called\n"; }

希望 wrapper(some_lvalue) 能调用 target(int&),而 wrapper(some_rvalue) 能调用 target(int&&)

c++ 复制代码
template<typename T>
void wrapper(T&& arg) { // arg在这里是"转发引用"
    // ...
    target(arg); // 问题出在这里!
}

int main() {
    int x = 10;
    wrapper(x);    // 期望调用 target(int&), 实际调用...
    wrapper(20);   // 期望调用 target(int&&), 实际调用...
}

C++有一个基本规则:任何有名字的变量,即使其类型是右值引用,它本身也是一个左值

  1. wrapper函数体内,arg有名字,所以它是一个左值
  2. target(arg)这行代码总是会调用target(int&),无论传给wrapper的是左值还是右值

完美转发 的目标就是解决这个问题:在转发参数时,完整地保留其原始的左值或右值属性

解决方案

std::forward

T里包含了原始类型信息,但函数体内的arg仍然是左值。这时就需要 std::forward`

std::forward 是一个有条件的类型转换 。它利用T中保存的类型信息,将arg恢复成其原始的值类别

std::forward<T>(arg) 的逻辑是:"检查模板参数T,如果T被推导为左值引用类型(如int&),则什么都不做,返回一个左值。如果T被推导为非引用类型(如int),则将arg转换为右值

c++ 复制代码
#include <iostream>
#include <utility> // for std::forward

void target(int& x) { std::cout << "lvalue reference called\n"; }
void target(int&& x) { std::cout << "rvalue reference called\n"; }

template<typename T>
void wrapper(T&& arg) {
    // 使用 std::forward 完美转发 arg
    target(std::forward<T>(arg));
}

int main() {
    int x = 10;
    std::cout << "Calling wrapper with lvalue:\n";
    wrapper(x); // T推导为int&, forward返回左值

    std::cout << "\nCalling wrapper with rvalue:\n";
    wrapper(20); // T推导为int, forward返回右值
}
sql 复制代码
Calling wrapper with lvalue:
lvalue reference called

Calling wrapper with rvalue:
rvalue reference called

特性 std::move std::forward
目的 转移所有权 保持值类别
行为 无条件转换为右值引用 有条件转换为右值引用
本质 static_cast<T&&>(arg) 基于模板参数 T 的条件转换
使用场景 当你确定一个对象不再需要,想 "窃取" 其资源时。 在泛型代码中,将参数以其原始值类别转发给另一个函数时。必须与转发引用一起使用。
相关推荐
Byron Loong11 小时前
【c++】为什么有了dll和.h,还需要包含lib
java·开发语言·c++
坚果派·白晓明11 小时前
【鸿蒙PC三方库移植适配框架解读系列】第一篇:Lycium C/C++ 三方库适配 — 概述与环境配置
c语言·开发语言·c++·harmonyos·开源鸿蒙·三方库·c/c++三方库
咩咦12 小时前
C++学习笔记02:cin 和 cout 输入输出
c++·学习笔记·cin·输入输出·cout
咩咦12 小时前
C++学习笔记05:引用和常引用
c++·学习笔记·引用·const·常引用
香蕉鼠片12 小时前
算法过程中不会的
开发语言·c++
阿旭超级学得完12 小时前
C++11包装器(function和bind)
java·开发语言·c++·算法·哈希算法·散列表
li星野12 小时前
位运算 & 数学 & 高频进阶九题通关(Python + C++)
c++·python·学习·算法
磊 子13 小时前
多态类原理+四种类型转换+异常处理
开发语言·c++·算法
王老师青少年编程13 小时前
csp信奥赛C++高频考点专项训练之字符串 --【回文字符串】:回文拼接
c++·字符串·csp·高频考点·信奥赛·字符串回文·回文拼接
Teleger15 小时前
在window上使用c++控制鼠标点击,实现的exe
c++·单片机·计算机外设