在编写 lambda 函数向量时出现隔离错误

Segfault while composing vector of lambda functions

本文关键字:隔离 错误 向量 lambda 函数      更新时间:2023-10-16

我正在尝试创建一个接受函数向量的compose_all lambda,并返回一个函数,该函数是向量中所有函数的组合:

#include <algorithm>
#include <iostream>
#include <vector>
#include <functional>
using std::cout;
using std::endl;
using std::function;
using std::vector;
int main() {
  vector<function<int(int)>> functions = {
    [](int x) { return 2 * x; },
    [](int x) { return x * x; },
    [](int x) { return -x; },
  };
  function<function<int(int)>(
      vector<function<int(int)>>::iterator,
      vector<function<int(int)>>::iterator,
      function<int(int)>)> compose_all;
  compose_all = [&](vector<function<int(int)>>::iterator f_begin,
                    vector<function<int(int)>>::iterator f_end,
                    function<int(int)> f) -> function<int(int)> {
    for (; f_begin < f_end; ++f_begin) {
      f = [&](int x) { return (*f_begin)(f(x)); };
    }
    return f;
  };
  auto composition = compose_all(functions.begin(),
                                 functions.end(),
                                 [](int x) { return x; });
  for (int i = 0; i < 10; ++i) {
    cout << composition(i) << endl;
  }
  return 0;
}

虽然这编译得很好,但它会出现段错误:

$ clang++ -std=c++11 -g composition.cpp && ./a.out 
Segmentation fault (core dumped)

段错误的原因是什么,解决方法是什么?

使用打印语句和 GDB 进行调试的说明:

  • 迭代器由compose_all正确接收
  • 段错误发生在线路上f = [&](int x) { return (*f_begin)(f(x)); };
  • 单独取消引用f_begin会产生正确的结果(它在向量中调用正确的 lambda)

创建的每个 lambda

f = [&](int x) { return (*f_begin)(f(x)); };

通过引用捕获ff_begin,其中两者都是存储在 compose_all 中的 lambda 主体的本地。

这些函数中的最后一个在调用时由compose_all包含的lambda的主体返回,然后分配给composition。但是由于compose_all的λ的主体已经退出,ff_begin的生命周期已经结束,调用composition是未定义的行为。

此外,你真的不希望f调用自己,这只会让你无限递归。您希望ff的当前值(即初始值或上次分配f值)调用 值的副本。

你需要这样的东西:

const auto& g = *f_begin;
f = [=](int x) { return g(f(x)); };

(或者在 C++14 或更高版本中,可以这样写:)

f = [f, g=*f_begin](int x) { return g(f(x)); };