C++ mutable关键字的使用场景_C++ lambda表达式修改值捕获变量

mutable关键字可使lambda表达式修改值捕获的变量。默认情况下,值捕获的变量在lambda中为const,不可修改;加上mutable后,允许修改其副本,如[=]() mutable { x++; }可正常编译并修改x的副本。

在C++中,mutable关键字主要用于突破const限制,特别是在lambda表达式和类成员函数中。它允许在原本不允许修改的上下文中修改变量。下面重点讲解它在lambda表达式中如何修改值捕获的变量。

lambda表达式中的值捕获与const默认行为

当lambda以值捕获方式([=])捕获外部变量时,这些变量在lambda内部默认被视为const,不能被修改。

例如:

int x = 10;
auto func = [=]() {
    // x++; // 错误:不能修改值捕获的变量
};
func();

此时编译会报错,因为lambda默认是const函数,不允许修改捕获的副本。

使用mutable关键字解除限制

在lambda表达式参数列表后加上mutable,就可以修改值捕获的变量。

修改上面的例子:

int x = 10;
auto func = [=]() mutable {
    x++; // OK:可以修改x的副本
    std::cout << x << std::endl; // 输出11
};
func();
std::cout << x << std::endl; // 外部x仍为10

注意:修改的是外部变量的副本,不会影响原始变量。

典型使用场景

  • 计数器或状态追踪:在多次调用lambda时维护内部状态
  • 延迟计算或缓存:第一次调用时初始化某些值
  • 算法中需要可变状态的函数对象:如std::for_each、std::generate等

示例:实现一个带计数功能的lambda

auto counter = [count = 0]() mutable {
    return ++count;
};
std::cout << counter() << std::endl; // 1
std::cout << counter() << std::endl; // 2

基本上就这些。mutable让lambda在保持值捕获的同时具备内部可变性,是一种安全且实用的机制。不复杂但容易忽略细节。