std::vector< MyObj* > to std::vector< MyObj Const * >,如何无循环?
std::vector< MyObj* > to std::vector< MyObj Const * >, how to no loops?
Title,
假设我们有
std::vector<MyObj*> ArrayOfPtr;
std::vector<MyObj const *> ArrayOfPtrToConstObj;
int main()
{
//I'd like to give ArrayOfPtr to another obj/function but not give it the right to modify the objects in it.
ArrayOfPtrToConstObj = ArrayOfPtr;
Function(ArrayOfPtrToConstObj)
}
为什么不能用C++编写并由编译器自动理解,考虑到 const 不会改变逻辑(优化或任何东西(..?
有没有一种快速的方法可以从向量 1 转到向量 2,而不必 for/while 循环到数组上并填充第二个?
好吧,某人,某个地方将不得不循环。您可以使用vector::assign
避免自己编写循环:
ArrayOfPtrToConstObj.assign(ArrayOfPtr.begin(), ArrayOfPtr.end());
但assign
仍然会循环往复。
首选的现代解决方案是传递一个 const-iterator/pointer 对并让用户使用它们,或者使用像 gsl::span
这样的视图类。这不需要复制任何内容,并且可以根据需要添加const
:
gsl::span<MyObj const*> spn(ArrayOfPtr.data(), ArrayOfPtr.size());
vector2
是一种根本不同的类型。 但是,如果您只想以const
的方式查看vector1
的内部,我为您提供了解决方案。
我会从类似但不完全是gsl::span
的东西开始。
template<class T>
struct span {
private:
T* b = 0;
T* e = 0;
public:
T* begin() const { return b; }
T* end() const { return e; }
T* data() const { return begin(); }
template<class U>
using compatible = std::enable_if_t< std::is_convertible< U*, T* >{} && sizeof(T)==sizeof(U), bool >;
template<class U,
compatible<U> =true
>
span( U* s, U* f ): b(s), e(f) {}
template<class U,
std::enable_if_t< std::is_convertible< U*, T* >{} && sizeof(T)==sizeof(U), bool > =true
>
span( U* s, std::size_t length ): span( s, s+length ) {}
// pointer semantics:
span()=default;
span(span const&)=default;
span& operator=(span const&)=default;
~span()=default;
std::size_t size() const{ return end()-begin(); }
bool empty() const{ return end()==begin(); }
T& front() const { return *begin(); }
T& back() const { return *(end()-1); }
// sub spans:
span without_front( std::size_t n=1 ) const {
n = (std::min)(n, size());
return {begin()+n, end()};
}
span without_back( std::size_t n=1 ) const {
n = (std::min)(n, size());
return {begin(), end()-n};
}
span only_front( std::size_t n=1 ) const {
n = (std::min)(n, size());
return {begin(), begin()+n};
}
span only_back( std::size_t n=1 ) const {
n = (std::min)(n, size());
return {end()-n, end()};
}
span sub( std::size_t start, std::size_t length ) const {
return without_front(start).only_front(length);
}
T& operator[](std::size_t I)const{ return begin()[I]; }
T& at(std::size_t I)const{
if (I>=size()) throw std::out_of_range{"index"};
return begin()[I];
}
std::vector<std::decay_t<T>> as_vector()const& {
return {begin(), end()};
}
std::vector<std::decay_t<T>> as_vector()&& {
return {std::make_move_iterator(begin()), std::make_move_iterator(end())};
}
template<class C,
compatible< std::decay_t< decltype( *std::declval<C&>().data() ) > > =true,
std::enable_if_t< !std::is_same<span, std::decay_t<C>>{}, bool > =true
>
span( C&& c ): span(c.data(), c.size()) {}
};
如果您不打算更改上下文中vector
的长度,则span
是正确的答案。
带有一些测试的现场示例。
在这种情况下,Function
应该采取span< MyObj const*const >
,说明它只想查看MyObj const
s的缓冲区。 如果需要,可以存储此类型的跨度,或者让隐式转换工作:
std::vector<MyObj*> ArrayOfPtr;
span<MyObj const*const> SpanOfPtrToConstObj;
int main()
{
//I'd like to give ArrayOfPtr to another obj/function but not give it the right to modify the objects in it.
SpanOfPtrToConstObj = ArrayOfPtr;
Function(SpanOfPtrToConstObj)
}
Function
无法修改矢量的长度或其中的指针或指针指向的内容。
相关文章:
- 使用std::multimap迭代器创建std::list
- C++中std::resize(n)和std::shrink_to_fit之间的区别
- 来自 std::list 的迭代器 .end() 按预期返回"0xcdcdcdcdcdcdcdcd"但 .begin()
- C++17复制构造函数,在std::unordereded_map上进行深度复制
- 如何导出包含具有"std::unique_ptr"值的"std::map"属性的
- 从持续时间构造std::chrono::system_clock::time_point
- std::具有相同基类的类的变体
- std::向量与传递值的动态数组
- 使用std::vector的OpenCL矩阵乘法
- std::map<struct,struct>::find 找不到匹配项,但是如果我循环通过 begin() 到 end(),我在那里看到匹配项
- std::condition_variable::wait()如何评估给定的谓词
- 如何获取std::result_of函数的返回类型
- std::原子加载和存储都需要吗
- 将对象移动到std::shared_ptr
- std::vector<;uint8_t>;当C++11/14启用时,手动复制而不是调用memcpy
- 是std :: set&lt; std :: future&gt;不可能存在
- 在修改列表后,std :: list&lt; t&gt; :: end()的值是否会更改
- 使用 std::vector<boost::shared_ptr<Base_Class>> 或 boost::p tr_vector 的性能注意事项是什么<Base>
- std::map<std::set, double> AND std:<long>map< std::p air<long, long>, double>
- 如何获取std::vector<DMatch>