c++11(二)

可变模板参数

cpp 复制代码
template<class ...ARGS>
void print(ARGS&&... args){
}

会自行生成多个参数的函数,如下面代码中可以自动识别有多少个参数。

cpp 复制代码
#include<iostream>
using namespace std;

template<class ...Args>
void Print(Args&&... args){
	cout<<sizeof...(args)<<endl;
} 
int main(){
	Print();
	Print(1);
	Print(1,"xxx");
	Print(1,"xxx",'?');
}

扩展包

可以打印参数

cpp 复制代码
void ShowList(){
	cout<<endl<<endl<<"fish"<<endl<<endl;
} 
template<class T,class ...Args>
void ShowList(T x,Args&&... args){
	cout<<x<<" ";
	ShowList(args...);
} 

lambda表达式

性质

C++ 中,Lambda 表达式本质上是一种"语法糖"。在编译阶段,编译器会自动生成一个匿名的类(闭包对象/Functor)。捕获列表中的变量会变成该类的私有成员变量(按值捕获即为成员拷贝,按引用捕获即为存储地址),而参数列表和函数体则会被转化为重载的 () 运算符。返回的函数名是UUID,保证每一次生成的函数名不相同。

使用

捕获列表 (参数列表) mutable(可选) 异常说明(可选) -> 返回值类型 { 函数体 }

捕捉的变量不可以修改,但是引用捕捉的对象可以修改

cpp 复制代码
#include <iostream>

int main() {
    int factor = 2; // 外部变量
    
    // 定义一个 Lambda:按值捕获 factor,接收一个 int 参数,返回 int
    auto multiply = [factor](int x) -> int {
        return x * factor;
    };

    std::cout << multiply(5) << std::endl; // 输出: 10
    return 0;
}
  • []:不捕获任何外部变量。
  • [=]:按值(复制一份副本)捕获所有被使用的外部局部变量。
  • [&]:按引用捕获所有被使用的外部局部变量。
  • [a, &b]:混合捕获,按值捕获 a,按引用捕获 b
  • [=, &a]:默认按值捕获所有变量,但 a 特例按引用捕获。
  • [this]:捕获当前对象的 this 指针,以便在 Lambda 中访问类的成员变量和函数。

包装器

1. Function(函数 / 函数包装器)

Function 本身意为"功能"或"数学函数",在编程中指执行特定任务的代码块。而在现代语言特性中,它通常指代通用的多态函数包装器 (如 C++ 的 std::function)。

  • 核心作用:统一类型。它可以存储、复制和调用任何可调用对象(如普通函数、Lambda表达式、仿函数等),将不同类型但签名相同的可调用对象统一成一个类型。
  • 典型应用场景:常用于实现回调函数、策略模式以及事件处理等需要解耦的场景。

2. Bind(参数绑定 / 适配器)

Bind 字面意思是"绑定"。在编程中,它是一个函数适配器 (如 C++ 的 std::bind 或 JavaScript 的 Function.prototype.bind())。

  • 核心作用:预先绑定函数的部分参数或上下文环境,从而生成一个新的、参数更少的可调用对象。
  • 典型应用场景
    • 参数绑定与柯里化:固定某些参数,简化后续调用。
    • 调整参数顺序 :通过占位符(如 _1, _2)改变原函数的参数传入顺序。
    • 绑定成员函数 :将类的成员函数转换为普通的可调用对象(通常需要绑定对象实例或 this 指针)。

使用

cpp 复制代码
// 定义两个普通的计算函数
int Add(int a, int b) { return a + b; }
int Multiply(int a, int b) { return a * b; }

// 使用 bind 提前绑定第一个参数为 5,生成新的可调用对象
auto addFive = std::bind(Add, 5, std::placeholders::_1);
auto mulTen = std::bind(Multiply, 10, std::placeholders::_1);

// 使用 function 统一包装,放入操作表中
std::map<std::string, std::function<int(int)>> operationTable;
operationTable["add_5"] = addFive;
operationTable["mul_10"] = mulTen;

// 实际调用:只需传入剩下的参数即可
std::cout << operationTable["add_5"](3);   // 输出: 8 (即 5 + 3)
std::cout << operationTable["mul_10"](4);  // 输出: 40 (即 10 * 4)
相关推荐
我不是外星人16 小时前
有了 Harness Engineering ,真的还需要研发工程师吗?
前端·后端·ai编程
IT_陈寒19 小时前
JavaScript的闭包把我坑惨了,说好的内存会自动回收呢?
前端·人工智能·后端
Jackson__20 小时前
分享一个横向滚动案例,带悬停暂停,通用性很强
前端
MariaH20 小时前
git rebase的使用
前端
_柳青杨20 小时前
深入理解 JavaScript 事件循环
前端·javascript
阡陌Jony20 小时前
关于前端性能优化的一些问题:
前端
用户6000718191021 小时前
【翻译】简化 TSRX
前端
IT乐手1 天前
佛德角逼平西班牙,国足还有啥借口?
前端
JustHappy1 天前
我汇总了身边朋友的经历才发现,其实第一份实习是最难找的......
前端·后端·面试
星栈1 天前
Dioxus 的响应式系统:`Signal`、`Memo`、`Effect` 和异步状态到底该怎么分工
前端·前端框架