卓越飞翔博客卓越飞翔博客

卓越飞翔 - 您值得收藏的技术分享站
技术文章75334本站已运行4311

C++ 匿名函数与函数对象的常见陷阱

匿名函数和函数对象在 c++++ 中存在陷阱,包括:1. 匿名函数捕获局部变量,导致值更改后仍使用捕获值;2. 函数对象长时间生命周期,可能造成内存泄漏;3. 函数对象交叉引用,引发循环引用。避免陷阱的最佳做法包括使用 [=]() 捕获所有局部变量、使用智能指针管理函数对象生命周期,以及避免函数对象之间的交叉引用。

C++ 匿名函数与函数对象的常见陷阱

C++ 匿名函数与函数对象的常见陷阱

简介

匿名函数和函数对象在 C++ 中非常有用,但它们也有一些常见陷阱需要注意。本文将介绍这些陷阱并提出避免它们的最佳做法。

立即学习“C++免费学习笔记(深入)”;

陷阱 1:匿名函数捕获局部变量

int main() {
  int x = 10;
  std::function<int()> fn = [&](){ return x; }; // 捕获 x
  x = 20;  // x 已更改
  return fn();  // 返回旧值 10
}

在上面的代码中,匿名函数 fn 捕获了局部变量 x。这意味着即使 x 的值在匿名函数外部发生变化,匿名函数仍将使用其捕获的值。这可能会导致微妙的错误。

最佳做法: 使用 [=]() 捕获所有局部变量,或者使用 const auto& 捕获局部变量的引用。

int main() {
  int x = 10;
  std::function<int()> fn = [=](){ return x; };  // 捕获所有局部变量
  x = 20;  // x 已更改
  return fn();  // 返回新值 20
}

陷阱 2:函数对象长时间生命周期

函数对象可以长时间存在,这可能会导致内存泄漏或其他问题。例如:

struct Foo {
  std::function<void()> fn;
  Foo() { fn = [](){ std::cout << "Hellon"; }; }
};

int main() {
  Foo foo;  // 函数对象在 foo 的整个生命周期内都存在
}

在上面的代码中,函数对象 fn 在 Foo 对象的整个生命周期内都存在。即使函数对象不再需要后,它仍会保存在内存中。

最佳做法: 确保函数对象只在需要时存在。一种方法是使用智能指针:

struct Foo {
  std::unique_ptr<std::function<void()>> fn;
  Foo() { fn = std::make_unique([](){ std::cout << "Hellon"; }); }
};

陷阱 3:函数对象交叉引用

函数对象之间可能存在交叉引用,这可能会导致循环引用并导致内存泄漏。例如:

struct Foo {
  std::function<void()> fn;
  Foo(std::function<void()>&& fn) : fn(fn) {}
};

struct Bar {
  std::unique_ptr<Foo> foo;
  Bar() { foo = std::make_unique<Foo>([&](){ bar->fn(); }); }
};

int main() {
  Bar bar;  // 循环引用
}

在上面的代码中,Foo 的构造函数捕获了 Bar 对象的 fn。Bar 的构造函数又创建了一个指向 Foo 的智能指针。这导致了 Foo 和 Bar 之间的循环引用,导致内存泄漏。

最佳做法: 避免函数对象之间出现交叉引用。一种方法是使用弱指针:

struct Foo {
  std::weak_ptr<Bar> bar;
  Foo(std::function<void()>&& fn) : fn(fn) {}
};

struct Bar {
  std::unique_ptr<Foo> foo;
  Bar() { foo = std::make_unique<Foo>([&](){ if (bar.lock()) bar.lock()->fn(); }); }
};
卓越飞翔博客
上一篇: PHP 函数中如何使用引用:理解引用生命周期
下一篇: 返回列表
留言与评论(共有 0 条评论)
   
验证码:
隐藏边栏