C++ 静态成员函数与 lambda 开销

Sym*_*ink 2 c++ lambda unique-ptr

我有某种模板化基类

template<typename Derived>
class Base { };
Run Code Online (Sandbox Code Playgroud)

并希望将其派生实例存储在列表中。为此,我使用using derived_handle = std::unique_ptr<void, void(*)(void*)别名。

当我现在将派生实例添加到列表中时,我可以使用静态成员函数作为删除器

class foo {

  template<typename Derived, typename... Args>
  void add_base(Args&&... args) {
    auto derived = derived_handle{new Base{std::forward<Args>(args)..., &foo::_deleter<Derived>};

    _derived.emplace_back(std::move(derived));
  }

private:

  template<typename Baser>
  void _deleter(void* base) {
    delete static_cast<Base*>(base);
  }

  std::vector<derived_handle> _derived{}; 

};
Run Code Online (Sandbox Code Playgroud)

或 lambda

class foo {

  template<typename Derived, typename... Args>
  void add_base(Args&&... args) {
    auto deleter = [](auto* derived){
      delete static_cast<Derived*>(derived);
    }

    auto derived = derived_handle{new Base{std::forward<Args>(args)..., std::move(deleter)};

    _derived.emplace_back(std::move(derived));
  }

private:

  std::vector<derived_handle> _derived{}; 

};
Run Code Online (Sandbox Code Playgroud)

我应该注意 lambda 版本的优点/缺点吗?

Sne*_*tel 5

是时候进行框架挑战了!

您在该代码中做出了一些错误的决定。大多数使用 的人unique_ptr,即使在多态上下文中,也根本不需要自定义删除器。你这样做的唯一原因是因为你的类型擦除,而这只是因为Base<A>Base<B>是不相关的类型。

如果您确实需要Base<T>,请让它从具有虚拟析构函数的实际多态(且非模板化)基类继承。那么你就不需要unique_ptr<void>(非常糟糕的代码味道),并且你实际上可以以类型安全的方式使用你的列表。