c – 将参数传递给“类似数组”的容器构造函数
背景
我正在使用嵌入式平台,但有以下限制: >没有堆 我过去曾经多次处理过以下问题:
该项目最近才增加了C 11支持,到目前为止,每次我必须处理这个问题时,我一直在使用临时解决方案.既然C 11可用,我想我会尝试制定一个更通用的解决方案. 解决方案尝试 我复制了an example of std::aligned_storage来为我的数组类型提出框架.结果如下: #include <type_traits> template<class T,size_t N> class Array { // Provide aligned storage for N objects of type T typename std::aligned_storage<sizeof(T),alignof(T)>::type data[N]; public: // Build N objects of type T in the aligned storage using default CTORs Array() { for(auto index = 0; index < N; ++index) new(data + index) T(); } const T& operator[](size_t pos) const { return *reinterpret_cast<const T*>(data + pos); } // Other methods consistent with std::array API go here }; 这是一种基本类型 – 阵列< T,N>如果T是默认可构造的,则仅编译.我对模板参数打包不是很熟悉,但是看一些例子让我得到以下结论: template<typename ...Args> Array(Args&&... args) { for(auto index = 0; index < N; ++index) new(data + index) T(args...); } 这绝对是朝着正确方向迈出的一步.阵列< T,N>现在编译,如果传递的参数匹配T的构造函数. 我唯一剩下的问题是构造一个数组< T,N>其中数组中的不同元素具有不同的构造函数参数.我想我可以把它分成两种情况: 1 – 用户指定参数 这是我对CTOR的抨击: template<typename U> Array(std::initializer_list<U> initializers) { // Need to handle mismatch in size between arg and array size_t index = 0; for(auto arg : initializers) { new(data + index) T(arg); index++; } } 除了需要处理数组和初始化列表之间的维度不匹配之外,这似乎工作正常,但有许多方法可以处理那些不重要的方法.这是一个例子: struct Foo { explicit Foo(int i) {} }; void bar() { // foos[0] == Foo(0) // foos[1] == Foo(1) // ..etc Array<Foo,10> foos {0,1,2,3,4,5,6,7,8,9}; } 2 – 参数遵循模式 在我之前的示例中,foos使用递增列表进行初始化,类似于std :: iota.理想情况下,我想支持以下内容,其中range(int)返回可以初始化数组的SOMETHING. // One of these should initialize foos with parameters returned by range(10) Array<Foo,10> foosA = range(10); Array<Foo,10> foosB {range(10)}; Array<Foo,10> foosC = {range(10)}; Array<Foo,10> foosD(range(10)); 谷歌搜索告诉我std :: initializer_list不是一个“普通”容器,所以我认为我没有办法让range(int)根据函数参数返回一个std :: initializer_list. 同样,这里有几个选项: >在运行时指定的参数(函数返回?) 问题 >到目前为止,此解决方案是否存在任何问题? 解决方法
如果我正确地理解你的问题,我也偶然发现了std :: array关于元素构造的完全不灵活性,有利于聚合初始化(以及缺少具有灵活元素构造选项的静态分配容器).我想出的最好的方法是创建一个类似自定义数组的容器,它接受一个迭代器来构造它的元素.
这是完全灵活的解决方案: >适用于固定尺寸和动态尺寸的容器 对于你的例子,它将是: const size_t SIZE = 10; std::array<int,SIZE> params; for (size_t c = 0; c < SIZE; c++) { params[c] = c; } Array<Foo,SIZE> foos(iterator_construct,¶ms[0]); //iterator_construct is a special tag to call specific constructor // also,we are able to pass a pointer as iterator,since it has both increment and dereference operators 注意:您可以通过使用自定义迭代器类完全跳过参数数组分配,该类从动态位置计算它的值. 对于多参数构造函数,它将是: const size_t SIZE = 10; std::array<std::tuple<int,float>,SIZE> params; // will call Foo(int,float) for (size_t c = 0; c < SIZE; c++) { params[c] = std::make_tuple(c,1.0f); } Array<Foo,piecewise_construct,¶ms[0]); 具体的实现示例是一个很大的代码,所以如果你想了解除了一般想法之外的实现细节的更多见解,请告诉我 – 我会更新我的答案. (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |