看起来如果不修改构造函数是不可能的boost::container::flat
。在不修改任何一个类的情况下,似乎只有 hack 可以做到这一点,例如使用reinterpret_cast
. 我找到的解决方案是使用替代实现vector
或非常丑陋的代码。
在进入我的解决方案之前,我必须说这可能是两个类的缺陷。这些类应该有一组
release()
/aquire(start, end)
函数,分别将指针范围返回给释放所有权的数据,并从那时起获取拥有它的指针范围。另一种方法是使用一个构造函数,该构造函数从具有数据成员函数的任何其他容器移动。
解决方案使用reinterpret_cast
和不同的实现vector
原来reinterpret_cast
ing from std::vector
toboost::container::flat_set
是不可能的,因为布局不兼容。但是,可以将 reinterpret_cast 从boost::container::vector
开箱boost::container::flat_set
即用(那是因为它们有一个共同的实现)。
#include<cassert>
#include<boost/container/flat_set.hpp>
int main(){
boost::container::vector<double> v = {1.,2.,3.};
boost::container::flat_set<double> fs = std::move(reinterpret_cast<boost::container::flat_set<double>&>(v));
assert(v.size() == 0);
assert(*fs.find(2.) == 2.);s
assert(fs.find(4.) == fs.end());
}
所以,我可以替换std::vector
为boost::container::vector
,我可以将数据移动到flat_set
.
std::vector
使用丑陋代码的非便携式解决方案
std::vector
和的布局boost::container::vector
不同的原因是boost::container::vector
元数据以这种方式存储:
class boost::container::vector{
pointer m_start;
size_type m_size;
size_type m_capacity;
}
而std::vector
(在 GCC 中)基本上是纯指针,
class std::vector{
pointer _M_start;
pointer _M_finish;
pointer _M_end_of_storage;
}
所以,我的结论是,鉴于我使用的实现std::vector
与boost::container::flat_set
.
在极端情况下,可以这样做(对不起,如果这段代码冒犯了某人,代码不可移植):
template<class T>
boost::container::flat_set<T> to_flat_set(std::vector<T>&& from){
// struct dummy_vector{T* start; T* finish; T* end_storarge;}&
// dfrom = reinterpret_cast<dummy_vector&>(from);
boost::container::flat_set<T> ret;
struct dummy_flat_set{T* start; std::size_t size; std::size_t capacity;}&
dret = reinterpret_cast<dummy_flat_set&>(ret);
dret = {from.data(), from.size(), from.capacity()};
// dfrom.start = dfrom.finish = dfrom.end_storarge = nullptr;
new (&from) std::vector<T>();
return ret;
};
int main(){
std::vector<double> v = {1.,2.,3.};
boost::container::flat_set<double> fs = to_flat_set(std::move(v));
assert(v.size() == 0);
assert(*fs.find(2.) == 2.);
assert(fs.find(4.) == fs.end());
}
请注意,我根本没有考虑分配器问题。我不确定如何在这里处理分配器。
回想起来,我不介意cast
对这个特定问题使用一种形式,因为我必须以某种方式告诉向量在移动到flat_set
. (问题是这变得极端,因为它是一个reinterpret_cast
。)但是这是一个次要问题,应该有合法的方式从 移动std::vector
到boost::container::vector
。