vector扩容时候的move问题
先抛出问题代码,下面代码的这种情况下存在一个内部指针泄露问题,伴随vector扩容,ItemManager的function内部存储的指针会指向错误的Item对象
因此,下面的这个Item类,虽然是move constructable的,但是实际上因为使用了lambda + capture了this指针,等价于将内部的this外传,item manager没有随之处理这种move的情况。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
#include <functional>
#include <iostream>
#include <map>
#include <vector>
using namespace std;
struct ItemManager {
void Register(int id, std::function<void()> &&func) {
funcs_[id] = std::move(func);
}
void Remove(int id) { funcs_.erase(id); }
std::map<int, std::function<void()>> funcs_;
};
ItemManager &GetItemManager() {
static ItemManager item_manager;
return item_manager;
}
struct Item {
int id;
Item(int id) : id(id) {
cout << "create item" << id << '\n';
GetItemManager().Register(
id, [this, id]() { printf("item %d, this=%p\n", id, this); });
}
~Item() {
cout << "destroy item" << id << '\n';
}
Item(Item &&item) : id(item.id) { cout << "move item" << id << '\n'; }
void print() const { printf("item %d, this=%p\n", id, this); }
};
int main() {
std::vector<Item> item_list;
auto print = [&item_list]() {
cout << "item_list print\n";
for (const auto &it : item_list) {
it.print();
}
cout << "function print\n";
for (const auto &[k, v] : GetItemManager().funcs_) {
v();
}
cout << '\n';
};
item_list.emplace_back(1);
print();
item_list.emplace_back(2);
print();
item_list.emplace_back(3);
print();
}
单独考虑扩容时候的情况,此时遇到cap的limit,会先开辟一片新的空间.
之前的老vec中的item会挪移到新开辟的vec中去,此时在新vec空间内的item,内部的this指针有了新的指向,外部manager内function所捕获的this指针没有更新,因此会指向错误的item对象。
下面的代码是vector扩容时候的emplace操作
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
template <class... _Valty>
_CONSTEXPR20 pointer _Emplace_reallocate(const pointer _Whereptr, _Valty&&... _Val) {
// reallocate and insert by perfectly forwarding _Val at _Whereptr
_Alty& _Al = _Getal();
auto& _My_data = _Mypair._Myval2;
pointer& _Myfirst = _My_data._Myfirst;
pointer& _Mylast = _My_data._Mylast;
_STL_INTERNAL_CHECK(_Mylast == _My_data._Myend); // check that we have no unused capacity
const auto _Whereoff = static_cast<size_type>(_Whereptr - _Myfirst);
const auto _Oldsize = static_cast<size_type>(_Mylast - _Myfirst);
if (_Oldsize == max_size()) {
_Xlength();
}
const size_type _Newsize = _Oldsize + 1;
size_type _Newcapacity = _Calculate_growth(_Newsize);
const pointer _Newvec = _STD _Allocate_at_least_helper(_Al, _Newcapacity);
const pointer _Constructed_last = _Newvec + _Whereoff + 1;
pointer _Constructed_first = _Constructed_last;
_TRY_BEGIN
_Alty_traits::construct(_Al, _STD _Unfancy(_Newvec + _Whereoff), _STD forward<_Valty>(_Val)...);
_Constructed_first = _Newvec + _Whereoff;
if (_Whereptr == _Mylast) { // at back, provide strong guarantee
if constexpr (is_nothrow_move_constructible_v<_Ty> || !is_copy_constructible_v<_Ty>) {
_STD _Uninitialized_move(_Myfirst, _Mylast, _Newvec, _Al);
} else {
_STD _Uninitialized_copy(_Myfirst, _Mylast, _Newvec, _Al);
}
} else { // provide basic guarantee
_STD _Uninitialized_move(_Myfirst, _Whereptr, _Newvec, _Al);
_Constructed_first = _Newvec;
_STD _Uninitialized_move(_Whereptr, _Mylast, _Newvec + _Whereoff + 1, _Al);
}
_CATCH_ALL
_STD _Destroy_range(_Constructed_first, _Constructed_last, _Al);
_Al.deallocate(_Newvec, _Newcapacity);
_RERAISE;
_CATCH_END
_Change_array(_Newvec, _Newsize, _Newcapacity);
return _Newvec + _Whereoff;
}
伴随_Newvec被分配,这里唯一的考虑_Uninitialized_move的情况,其余类似,内存会被腾挪到新开辟的array里
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
template <class _InIt, class _Alloc>
_CONSTEXPR20 _Alloc_ptr_t<_Alloc> _Uninitialized_move(
const _InIt _First, const _InIt _Last, _Alloc_ptr_t<_Alloc> _Dest, _Alloc& _Al) {
// move [_First, _Last) to raw _Dest, using _Al
// note: only called internally from elsewhere in the STL
using _Ptrval = typename _Alloc::value_type*;
auto _UFirst = _STD _Get_unwrapped(_First);
const auto _ULast = _STD _Get_unwrapped(_Last);
if constexpr (conjunction_v<bool_constant<_Iter_move_cat<decltype(_UFirst), _Ptrval>::_Bitcopy_constructible>,
_Uses_default_construct<_Alloc, _Ptrval, decltype(_STD move(*_UFirst))>>) {
#if _HAS_CXX20
if (!_STD is_constant_evaluated())
#endif // _HAS_CXX20
{
_STD _Copy_memmove(_UFirst, _ULast, _STD _Unfancy(_Dest));
return _Dest + (_ULast - _UFirst);
}
}
_Uninitialized_backout_al<_Alloc> _Backout{_Dest, _Al};
for (; _UFirst != _ULast; ++_UFirst) {
_Backout._Emplace_back(_STD move(*_UFirst));
}
return _Backout._Release();
}
最后,move完成了,摧毁老的array
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
_CONSTEXPR20 void _Change_array(const pointer _Newvec, const size_type _Newsize, const size_type _Newcapacity) {
// orphan all iterators, discard old array, acquire new array
auto& _Al = _Getal();
auto& _My_data = _Mypair._Myval2;
pointer& _Myfirst = _My_data._Myfirst;
pointer& _Mylast = _My_data._Mylast;
pointer& _Myend = _My_data._Myend;
_My_data._Orphan_all();
if (_Myfirst) { // destroy and deallocate old array
_STD _Destroy_range(_Myfirst, _Mylast, _Al);
_ASAN_VECTOR_REMOVE;
_Al.deallocate(_Myfirst, static_cast<size_type>(_Myend - _Myfirst));
}
_Myfirst = _Newvec;
_Mylast = _Newvec + _Newsize;
_Myend = _Newvec + _Newcapacity;
_ASAN_VECTOR_CREATE;
}
因此,任意对象,只要存在内部资源泄露(指针或非POD对象)问题,且自定义的move函数没有对过期资源的更新操作,这个对象其实是不能放进内存连续的顺序容器的
更求稳,能放进容器的对象,尽量是POD的。
删除头部元素,后面元素的move + 摧毁旧的item的操作同理,相似
判断类是不是可移动的
1
2
3
4
5
6
7
template <typename T>
void check_if_moveable() {
std::cout << "Is move constructible: " << std::boolalpha
<< std::is_move_constructible<T>::value << std::endl;
std::cout << "Is move assignable: " << std::is_move_assignable<T>::value
<< std::endl;
}
c++的标准在不同编译器 + 版本上一直在变,这个实际上要自己确认一下具体对象的情况,不过这个函数可以作为一个参考
This post is licensed under CC BY 4.0 by the author.