1

我不确定标题,因为我不确定问题来自容器的“可复制性”。我尝试了一切,但我无法摆脱这个错误。

这是我的代码的简化版本(请不要挑战类设计,我真的很想将最终使用的语法保留在 BOOST_FOREACH 中):

template <typename T>
class MyContainer
{
public:
    typedef typename std::vector<T>::iterator iterator;
    typedef typename std::vector<T>::const_iterator const_iterator;

    MyContainer(std::vector<T>& vec, boost::mutex& mutex) :
        m_vector(vec),
        m_lock(mutex)
    {
    }

    iterator begin() { return m_vector.begin(); }
    const_iterator begin() const { return m_vector.begin(); }
    iterator end() { return m_vector.end(); }
    const_iterator end() const { return m_vector.end(); }


private:
    std::vector<T>& m_vector;
    boost::lock_guard<boost::mutex> m_lock;
};

template <typename T>
struct GetContainer
{
    GetContainer(std::vector<T>& vec, boost::mutex& mutex) :
        m_vector(vec),
        m_mutex(mutex)
    {
    }

    MyContainer<T> Get()
    {
        return MyContainer<T>(m_vector, m_mutex);
    }

    std::vector<T>& m_vector;
    boost::mutex& m_mutex;
};



int main()
{
    std::vector<int> v;
    v.push_back(1);
    v.push_back(2);
    boost::mutex m;

    GetContainer<int> getter(v, m);

    BOOST_FOREACH(int i, getter.Get())
    {
        std::cout << i << std::endl;
    }

    return 0;
}

编译器抱怨没有 MyContainer::MyContainer(const MyContainer&) 的复制构造函数。我也有:错误:没有匹配函数调用'MyContainer::MyContainer(boost::foreach_detail_::rvalue_probe >::value_type)'</p>

我遵循可扩展性提示: http: //www.boost.org/doc/libs/1_58_0/doc/html/foreach/extensibility.html#foreach.extensibility.making__literal_boost_foreach__literal__work_with_non_copyable_sequence_types

但是,使

MyContainer<T> : private boost::noncopyable

不能解决问题。也不定义函数

boost_foreach_is_noncopyable

或专门化模板结构

is_noncopyable

对于 MyContainer (事实上,我如何将这个模板专门用于模板类型?)

最后一个“提示”:如果我从任何地方删除互斥锁和锁(我只是将向量传递给 GetContainer 和 MyContainer),它就可以工作。但如果我做它不起作用

MyContainer<T> : private boost::noncopyable

(我希望它应该如此,所以我不确定我的问题出在 BOOST_FOREACH 上,但可能是因为我用 getter 返回了 MyContainer 的副本?)

如果您在此处阅读我,我将感谢您,并提前感谢您的帮助。

4

2 回答 2

1

似乎是仅限移动类型的 BOOST_FOREACH 的限制。我没有找到解决方法¹(除了 - 丑陋 - 明显的方法将lock_guarda放入shared_ptr)。

但是,您没有指定 c++03 要求,因此您可以通过替换lock_guardunique_lock.

这是我对 c++11 的看法(注意它的通用性):

Live On Coliru

#include <boost/thread.hpp>
#include <boost/range.hpp>

namespace detail {
    template <typename R, typename M>
    struct RangeLock {
        RangeLock(R&r, M& m) : _r(r), _l(m) {}
        RangeLock(RangeLock&&) = default;

        using iterator = typename boost::range_iterator<R>::type;
        iterator begin() { using std::begin; return begin(_r); }
        iterator end  () { using std::end;   return end  (_r); }

        using const_iterator = typename boost::range_iterator<R const>::type;
        const_iterator begin() const { using std::begin; return begin(_r); }
        const_iterator end  () const { using std::end;   return end  (_r); }

     private:
        R& _r;
        boost::unique_lock<M> _l;
    };
}

template <typename R, typename M>
    detail::RangeLock<R,M> make_range_lock(R& r, M& mx) { return {r,mx}; }
template <typename R, typename M>
    detail::RangeLock<R const,M> make_range_lock(R const& r, M& mx) { return {r,mx}; }

#include <vector>
#include <map>

int main() {

    boost::mutex mx;

    std::vector<int> const vec { 1, 2 };
    std::map<int, std::string> const map { { 1, "one" }, { 2, "two" } };

    for(int i : make_range_lock(vec, mx))
        std::cout << i << std::endl;

    for(auto& p : make_range_lock(map, mx))
        std::cout << p.second << std::endl;

    for(auto& p : make_range_lock(boost::make_iterator_range(map.equal_range(1)), mx))
        std::cout << p.second << std::endl;

}

印刷

1
2
one
two
one

¹ 甚至没有使用Using BOOST_FOREACH 中的所有方法和一个恒定的侵入性列表

于 2015-05-19T21:22:50.010 回答
0

如果有帮助,我会发布我的答案...

使用 C++03,我终于提供了一个复制构造函数,以便能够将类与 BOOST_FOREACH 一起使用。所以问题转移到另一个话题:使类以逻辑和适当的方式复制。

就我而言,我“共享锁和向量”,如果用户不想做错误,则不应使用此副本本身,但在 BOOST_FOREACH 中没关系:

  • 我将互斥锁更改为 recursive_mutex
  • 我将锁更改为 unique_lock 并:

    MyContainer(const MyContainer& other) :
                                m_vector(other.vec),
                                m_lock(*other.m_lock.mutex())
    {
    }
    

使用 C++11

感谢 boost 邮件列表中的 Chris Glover,这是一个 C++11 解决方案:

你不能在 C++03 中做你想做的事情。要完成它,您需要 C++11 移动语义才能将 MyContainer 移出 Get 函数。即使不使用 BOOST_FOREACH,以下代码也会失败;

GetContainer<int> getter(v, m); 
MyContainer<int> c = getter.Get(); // <-- Error. 

这是一个进行必要更改的示例;我将 scoped_lock 更改为 unique_lock 并添加了一个移动构造函数。

template <typename T> 
class MyContainer 
{ 
public: 
[...]
    MyContainer(MyContainer&& other) 
        : m_vector(other.m_vector) 
    { 
        m_lock = std::move(other.m_lock); 
        other.m_vector = nullptr; 
    } 
于 2015-05-21T08:28:33.573 回答