当前位置: 首页 > news >正文

c++bind和forward完美转化


前言

1. std::bind概述

std::bind是C++11引入的功能模板,位于<functional>头文件中,用于将函数、成员函数或函数对象与特定参数绑定,生成一个新的可调用对象。

1.1 基本用法

#include <iostream>
#include <functional>void print_sum(int a, int b) {std::cout << a + b << std::endl;
}int main() {// 绑定函数和参数auto bound_print = std::bind(print_sum, 10, std::placeholders::_1);bound_print(20);  // 输出30,相当于print_sum(10, 20)return 0;
}

std::bind 将函数 print_sum 和部分参数绑定,生成一个新的可调用对象 bound_printstd::placeholders::_1 表示调用 bound_print 时的第一个参数会传递给 print_sum 的第二个形参 b。绑定时已固定的参数(如 10)在调用时不再需要传入。

 默认行为:按值捕获参数

int x = 10;
auto bound = std::bind(f, x);  // x被复制
x = 20;
bound();  // 使用x的副本(值为10)

引用捕获:使用std::ref或std::cref

int x = 10;
auto bound = std::bind(f, std::ref(x));  // x被引用捕获
x = 20;
bound();  // 使用x的引用(值为20)

1.2 占位符

std::placeholders::_1_2, ..., _N表示调用时传入的第1, 2, ..., N个参数

void print_values(int a, int b, int c) {std::cout << a << ", " << b << ", " << c << std::endl;
}int main() {auto bound_func = std::bind(print_values, std::placeholders::_2,std::placeholders::_1,100);bound_func(10, 20);  // 输出20, 10, 100return 0;
}

调用 bound_func(10, 20) 时: _1 对应第一个实参 10 → 传给原函数的 b   _2 对应第二个实参 20 → 传给原函数的 a  c 始终为固定值 100

1.3 绑定成员函数

绑定成员函数时需要传入对象指针或引用:

class MyClass {
public:void print(int x) {std::cout << "Value: " << x << std::endl;}
};int main() {MyClass obj;auto bound_member = std::bind(&MyClass::print, &obj, std::placeholders::_1);bound_member(42);  // 输出Value: 42return 0;
}
成员函数指针&MyClass::print 是 MyClass 的成员函数 print 的指针。在 C++ 中,成员函数指针必须通过类的对象(或指针)来调用。std::bind 绑定成员函数std::bind 的第一个参数是成员函数指针,第二个参数是对象的指针(&obj),后续参数是成员函数的参数(用占位符或固定值)。 占位符 _1std::placeholders::_1 表示调用 bound_member 时的第一个参数会传递给 print 的 x 调用方式bound_member(42)

2. 完美转发(Perfect Forwarding)

完美转发是指在函数模板中将参数以原始类型转发给另一个函数,保持参数的值类别(左值、右值)不变。

2.1 引用折叠规则

完美转发基于引用折叠规则:

  • T& & → T&   T& && → T&    T&& & → T&   T&& && → T&&

2.2 完美转发示例

#include <utility>void process(int& x) { std::cout << "lvalue: " << x << std::endl; }
void process(int&& x) { std::cout << "rvalue: " << x << std::endl; }template<typename T>
void wrapper(T&& arg) {process(std::forward<T>(arg));  // 完美转发
}int main() {int x = 10;wrapper(x);       // 调用lvalue版本wrapper(20);      // 调用rvalue版本wrapper(std::move(x)); // 调用rvalue版本return 0;
}

 wrapper(x)(左值)x 是左值(有名变量),T 推导为 int&std::forward<int&>(arg) 返回左值引用。调用 process(int&),输出 lvalue: 10

wrapper(20)(右值)20 是右值(临时值),T 推导为 intstd::forward<int>(arg) 返回右值引用。调用 process(int&&),输出 rvalue: 20

 wrapper(std::move(x))(右值)std::move(x) 将 x 转为右值,T 推导为 intstd::forward<int>(arg) 返回右值引用。调用 process(int&&),输出 rvalue: 10

3. 注意事项

只在模板函数中使用   转发后不要再用该参数、不要重复转发同一个参数

记住:T&& + forward<T> 就是完美转发的全部秘诀


总结

  1. std::bind用于部分应用和参数重排序    完美转发保持参数的值类别

  2. 现代C++中优先考虑lambda表达式   需要完美转发时,确保正确使用std::forward

通过理解这些概念,可以编写更灵活、高效的C++代码,特别是在涉及回调、延迟调用和泛型编程的场景中。

相关文章:

  • 实现 “WebView2 获取word选中内容
  • [NocoDB] 在局域网中调整Float类型显示精度的部署经验
  • 【笔记】在Cygwin上使用mintty连接wsl
  • DeepLegal AI:智能法律文档审查与合规助手+MVP
  • 保存 QTextEdit 内容打包成一个文件(包含文本和图片)
  • 提示词模板设计:LangGPT的提示词设计框架
  • 《深度解析:如何打造高性能短剧平台?完整技术方案与行业实践》
  • 深入理解PHP中的面向对象编程
  • C3新增特性
  • ps外发光
  • Flink维表应用:从思考到实践的全面解析
  • Vue 中 filter 过滤的语法详解与注意事项
  • 项目上线(若依前后分离版)
  • ganymed-ssh2连接openssh 8.2
  • 没有产品说明书和需求文档的情况下能够进行黑盒测试吗?
  • 黑马python(十五)
  • Python异步爬虫编程技巧:从入门到高级实战指南
  • 爬虫002-----urllib标准库
  • 【GNSS软件接收机】【理论简介】Chapter.3 RAIM 和 FDE[2025年6月]
  • QML革命:下一代GUI开发的核心优势详解
  • 网站备案照片 多少钱/网文推广怎么做
  • 中文域名.网站/南京网络优化公司有哪些
  • 设计类赚钱网站/电商平台哪个最好最可靠
  • 网站搭建行业/关键词分类工具
  • 国内酷炫网站/赛事资讯赛马资料
  • 建设网站前的市场分析包括哪些内容/怎么做市场推广