使用迭代器访问 vector<std::unique_ptr > 的元素<T>?
Access elements of vector<std::unique_ptr<T> > using an iterator?
我有一个类成员变量
vector<std::unique_ptr<T> > v;
和一个成员函数,其中我想使用v
的unique_ptr
元素,由iterator
参数"寻址"。哪一个更好?
void mem_fun(vector<std::unique_ptr<T> >::iterator it) {
std::unique_ptr<T> p;
p = std::move(*it);
...
}
或
void mem_fun(vector<std::unique_ptr<T> >::iterator it) {
std::unique_ptr<T>& p = *it;
...
}
据我所知,第二种方式似乎有点违反了unique_ptr
的"唯一性"。但是std::move()
能移动*it
吗?BTW,谁真正拥有unique_ptr
指针、类、成员向量、任何成员函数,或者其他什么?
Herb Sutter的这篇文章包含了回答您问题的知识。
unique_ptr
实际上定义了一个"所有权证书"。此所有权是不可复制的。当您move
和unique_ptr
时,这将有效地销毁存储在"vector"中的证书。
因此,vector<unique_ptr<T>>
拥有T
s。
当您只想用T
做一些事情时,您应该将您的函数声明为
void mem_fun(T& t) { ... }
并将取消引用委托给另一个函数:
template<typename It, typename F>
void foreach_deref(It from, It to, F f) {
std::for_each(from, to, [&](decltype(*from) &pt) {
f(*pt);
}
}
并使用它调用您的成员函数:
foreach_deref(begin(v), end(v), [&](T& t) { mem_fun(t); });
第一个版本拥有unique_ptr
目标的所有权,并将向量留给空的unique_ptr
s。这不太可能是您想要的。第二个版本令人困惑。如果您只想在不影响所有权的情况下访问unique_ptr
管理的对象,只需使用引用运算符:
(*it)->someMethodOfT();
这里,去引用(*it)
是对迭代器的去引用,而->
是对unique_ptr
的去引用。
记住:unique_ptr
的"唯一性"是指其所有权。没有什么可以说托管对象不能被许多非所有者访问。但是,由您决定谁拥有所有权,这取决于您的应用程序的要求。
p = std::move(*it);
这意味着,p
现在拥有*it
以前拥有的东西,而*it
不再拥有它(*(。
事实上,此后使用*it
可能会导致未定义的行为,因为它已从中移出。
*it
后面的unique_ptr
由矢量所有。所以实际上你改变了向量。我认为这不是你想要的,所以最好使用参考版本。
[…]第二种方式似乎有点违反了"唯一性"[…]
在这种情况下,唯一性意味着对象的单个所有者。当然,您可能有多个对该对象所有者(即指针(的引用。
(*(这也意味着当p
超出作用域时,对象将被销毁和释放。(除非您从p
移动到某个地方(
- 请解释这句话(cout<<1+int((a<b)^((b-a)&1) )<<endl
- 呼叫运营商<<临时
- 如何防止clang格式在流运算符调用之间添加换行符<<
- <<操作员在下面的行中工作
- EASTL矢量<向量<int>>连续的
- C - 创建矢量&lt; vector&lt; double&gt;&gt;矩阵具有分配而不是inizializ
- 为什么将此对向量&lt; map&lt; int,int&gt;&gt;中的地图进行更新.失败
- C :对矢量进行排序&lt; struct&gt;(结构有2个整数)基于结构的整数之一
- 明确的专业化“ CheckIntmap&lt;&gt;”实例化
- 什么是模板&lt;&gt;inline bla bla
- 编辑C Qlist&lt; object*&gt; gt;QML代码和一些QML警告中的模型
- eigen :: llt&lt;eigen :: matrixxd&gt;具有不完整的类型
- 错误,包括&lt; ctype&gt;在原子上使用C 11
- std::vector<;uint8_t>;当C++11/14启用时,手动复制而不是调用memcpy
- 如何加入向量&lt; int&gt;到C 中的单个INT
- 是std :: set&lt; std :: future&gt;不可能存在
- 是numeric_limits&lt; int&gt; :: is_modulo从逻辑上矛盾
- opencv 2.4.7在iOS错误背景_segm.hpp #include&lt; list&gt;未找到
- 在修改列表后,std :: list&lt; t&gt; :: end()的值是否会更改
- ///<评论></评论>在Visual Studio中