如何将 std::find_if 与唯一指针向量一起使用?

How to use std::find_if with a vector of unique pointers?

如何将 std::find_if 之类的算法与唯一指针向量一起使用?例如:

#include <iostream>
#include <vector>
#include <memory>
#include <algorithm>

class Integer {
public:
    explicit Integer(int i): i_(i){};

    int get() const{
        return i_;
    }

private:
    int i_;
};

using IntegerPtr = std::unique_ptr<Integer>;

int main() {
    IntegerPtr p1 = std::make_unique<Integer>(4);
    IntegerPtr p2 = std::make_unique<Integer>(5);
    IntegerPtr p3 = std::make_unique<Integer>(6);
    std::vector<IntegerPtr> vectorOfIntegerPointers({
        std::move(p1),
        std::move(p2),
        std::move(p3),
    });

    int i = 5;

    auto first_index_larger_than_i = std::find_if(vectorOfIntegerPointers.begin(), vectorOfIntegerPointers.end(), [&](IntegerPtr s) {
        return s->get() > i;
    });

    std::cout << first_index_larger_than_i.get() << std::endl;

    return 0;
}

失败

/Applications/Xcode.app/Contents/Developer/Toolchains/XcodeDefault.xctoolchain/usr/bin/../include/c++/v1/memory:1881:31: error: call to implicitly-deleted copy constructor of 'std::__1::unique_ptr<Integer, std::__1::default_delete<Integer> >'
            ::new((void*)__p) _Up(_VSTD::forward<_Args>(__args)...);
                              ^   ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

代码中有 2 个问题都导致尝试复制不可复制的 unique_ptr

  1. unique_ptr 无法传递给 vector(initializer_list) 构造函数,因为 initializer_list 将其元素包装为 const 对象,并且 const 无法移动对象。所以移动构造函数不参与重载决议,只留下复制构造函数作为候选,后来编译失败并出现您看到的错误:“调用隐式删除的复制构造函数".

    所以你必须使用另一种解决方案来构造vector<unique_ptr>,例如使用push_back:

     std::vector<IntegerPtr> vectorOfIntegerPointers;
     vectorOfIntegerPointers.push_back(std::make_unique<Integer>(4));
     vectorOfIntegerPointers.push_back(std::make_unique<Integer>(5));
     vectorOfIntegerPointers.push_back(std::make_unique<Integer>(6));
    

    或者编写一个包装器将 unique_ptr 保存为 mutable 成员 ()。

  2. [&](IntegerPtr s) { ... } 用于 std::find_if 尝试按值获取 unique_ptr 的实例。但是 unique_ptr 不可复制,因此出现同样的错误。

    一个快速的解决方法是通过引用代替它们:

    [&](IntegerPtr const& s) { ... }