如何将具有unique_ptr向量的类克隆到基类

时间:2018-12-13 14:49:49

标签: c++

#include <iostream>
#include <memory>
#include <vector>
#include <utility>
#include <map>

class IOutput{
public:
    virtual std::ostream& print(std::ostream& out) const = 0;
    virtual std::unique_ptr<IOutput> clone() const = 0;
    virtual ~IOutput() = default;
};

std::ostream& operator<<(std::ostream& out, const IOutput& ser) {
    ser.print(out);
    return out;
}

class String : public IOutput {
private:
    std::string m_str;
public:
    String( std::string str )
        : m_str(std::move(str))
    {}
    std::ostream& print(std::ostream& out) const override {
        out << m_str;
        return out;
    }
    std::unique_ptr<IOutput> clone() const override { return std::make_unique<String>(*this); }
};

class List : public IOutput {
private:
    std::vector<std::unique_ptr<IOutput>> m_elements;
public:
    std::ostream& print(std::ostream& out) const override {

    }
    std::unique_ptr<IOutput> clone() const override {
        return std::make_unique<List>(*this); 
    } 
};




int main () {
    String s {"string s"};
    String s2 {"string s2"};
    String s3 {"string s3"};
    String s4 {"string s4"};


    return 0;
}

错误

/usr/include/c++/7/bits/stl_construct.h:75: error: use of deleted function ‘std::unique_ptr<_Tp, _Dp>::unique_ptr(const std::unique_ptr<_Tp, _Dp>&) [with _Tp = IOutput; _Dp = std::default_delete<IOutput>]’
 { ::new(static_cast<void*>(__p)) _T1(std::forward<_Args>(__args)...); }
   ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~

我发现只有在std::vector<std::unique_ptr<IOutput>>中有List时才会发生此错误。看来我在 clone 中调用std::make_unique<List>(*this)时正在复制该向量,这意味着它必须复制该向量中的每个元素,在我们的情况下(unique_ptr)被禁止。 br /> 如何解决呢?

3 个答案:

答案 0 :(得分:6)

编译器无法生成副本构造函数,因为矢量成员不可复制。如果希望支持复制,则需要自己实现,例如:

List(List const& other) : m_elements(other.m_elements.size()) {
    std::transform(begin(other.m_elements), end(other.m_elements), begin(m_elements),
      [](std::unique_ptr<IOutput> const& o_ptr) {
        return o_ptr->clone();
      }
    );
}

答案 1 :(得分:1)

没有std::transform

的替代解决方案
List(const List& other)
{
    m_elements.reserve(other.m_elements.size());
    for (const auto& x: other.m_elements) {
        m_elements.emplace_back(x->clone());
    }
}

答案 2 :(得分:0)

IOutput具有clone()成员函数。您应该制作一个新的List,其中应包含向量元素的克隆。