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 的条件转换
使用场景 当你确定一个对象不再需要,想 "窃取" 其资源时。 在泛型代码中,将参数以其原始值类别转发给另一个函数时。必须与转发引用一起使用。
相关推荐
Wave8458 小时前
C++继承详解
开发语言·c++·算法
Tairitsu_H8 小时前
C++类基础概念:定义、实例化和this指针
开发语言·c++
不想写代码的星星8 小时前
C++17 string_view 观察报告:好用,但有点费命
c++
努力努力再努力wz8 小时前
【Linux网络系列】深入理解 I/O 多路复用:从 select 痛点到 poll 高并发服务器落地,基于 Poll、智能指针与非阻塞 I/O与线程池手写一个高性能 HTTP 服务器!(附源码)
java·linux·运维·服务器·c语言·c++·python
努力努力再努力wz8 小时前
【Linux网络系列】万字硬核解析网络层核心:IP协议到IP 分片重组、NAT技术及 RIP/OSPF 动态路由全景
java·linux·运维·服务器·数据结构·c++·python
minji...9 小时前
Linux 线程同步与互斥(四) POSIX信号量,基于环形队列的生产者消费者模型
linux·运维·服务器·c语言·开发语言·c++
王老师青少年编程9 小时前
csp信奥赛C++高频考点专项训练之贪心算法 --【排序贪心】:拼数
c++·算法·贪心·csp·信奥赛·排序贪心·拼数
程序猿编码9 小时前
给Linux程序穿“隐身衣”——ELF运行时加密器全解析(C/C++代码实现)
linux·c语言·c++·网络安全·elf·内存安全
John_ToDebug9 小时前
从 Win10 到 Win11 22H2+:任务栏美化中的“蒙版”和“Hover 色块”渲染原理解析
c++·chrome·windows
谭欣辰10 小时前
AC自动机:多模式匹配的高效利器
数据结构·c++·算法