- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我有一个自定义的 Vec 类,复制了 std::vector 的功能,但我无法实现一个删除函数,它采用与标准库实现相同的参数(并优雅地处理它们)。具体来说,在 C++11 中,vector::erase具有签名iterator erase (const_iterator position);
,其中返回的迭代器指向元素删除后元素的新位置。我唯一的解决方案是传递一个非常量迭代器,用第二个迭代器将给定迭代器之后的所有元素复制回一个位置,并使用第三个迭代器存储原始指针位置。这需要三个非常量迭代器。
template <class T> typename Vec<T>::iterator Vec<T>::erase(iterator it)
{
iterator copy_iterator = it; // used to shift elements one position to the left past the deleted element.
iterator return_iterator = it; // holds original it position, per std implementation
while (it != (this -> end() - 1)) { // copies elements one position left
*copy_iterator++ = *++it;
}
alloc.destroy(it); // destroy last element in vector
avail = it; // shortens the vector by 1
return return_iterator;
}
这里的 avail 是一个迭代器,它指向一个 vector 的末尾,即 iterator end() { return avail; }
。如果它必须将每个元素左移一个,我不明白任何这样的函数怎么可能采用 const_iterator,而且我真的不喜欢有三个迭代器。有更好的解决方案吗?
附加标准问题:
Up until C++98 , vector::erase 接受了一个迭代器参数。我很好奇为什么要更改此标准。在 C++11 中,删除函数包括看起来从 const_iterator 到迭代器的直接转换,而没有进一步解释为什么它现在是 const。
template <class T> typename vector<T>::iterator erase(const_iterator __position) {
difference_type __ps = __position - cbegin();
pointer __p = this->__begin_ + __ps;
iterator __r = __make_iter(__p);
this->__destruct_at_end(_VSTD::move(__p + 1, this->__end_, __p));
return __r;
}
这是 Vec 类的部分实现:
template <class T> class Vec {
public:
// typedefs
Vec() { create(); }
explicit Vec(std::size_t n, const T& val = T()) { create(n, val); }
Vec(const Vec& v) { create(v.begin(), v.end()); } // copy constructor
Vec& operator=(const Vec&);
~Vec() { uncreate(); } // destructor
T& operator[](size_type i) { return data[i]; }
const T& operator[](size_type i) const { return data[i]; }
void push_back(const T& val) {
if (avail == limit)
grow();
unchecked_append(val);
}
size_type size() const { return avail - data; }
iterator begin() { return data; }
const_iterator begin() const { return data; }
iterator end() { return avail; }
const_iterator end() const { return avail; }
iterator erase(iterator);
private:
iterator data;
iterator avail;
iterator limit;
allocator<T> alloc;
void create();
void create(size_type, const T&);
void create(const_iterator, const_iterator); // calls alloc.allocate, then copies without default initialization
void uncreate(); // calls alloc.destroy on everything, then deallocates
void grow(); // allocates twice the memory
void unchecked_append(const T&);
};
最佳答案
只需将您的const_iterator
转换为函数内的iterator
。现在,通常情况下,这是一个禁忌。但是因为我们知道 Vec
不是 const(因为 erase
是一个非常量函数),这是安全的,除非用户传递了一个不属于的迭代器到 Vec
(无论如何都不安全)。
那么,如何从 const_iterator
得到一个非常量 iterator
呢?您可以在 const_iterator
类中实现一个私有(private)成员来执行此操作并使 Vec
类成为友元。但由于我们正在处理随机访问迭代器,因此没有必要这样做,因为我们可以只使用迭代器算法。
// given a const_iterator cit
iterator it = begin() + (cit - begin());
关于c++ - std::erase 如何为 vector 实现?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45114064/
假设您要按值从 vector 中删除单个元素。 remove 之间有什么区别? -删除: vector v; // add some values vector::iterator it = remo
class CSensor { public: CSensor(int nVal1,char* pVal2,unsigned int nVal3); CSensor(cons
在回答过程中another question我偶然发现 std::vector::erase() 和 std::deque::erase() 的措辞略有不同。 这就是 C++14 关于 std::de
我想处理 vector 中的元素一段时间。为了优化这一点,我不想在处理项目时删除它,而是在最后删除所有已处理的项目。 vector::iterator it; for(it = items.begin
是否 std::set erase ( x ) 在 STL 中以相同的方式实现 erase ( collection.find( x ) ); (其中 x 是 const key_type&) 吗?
This question already has answers here: How to catch .erase exception in vectors? (2个答案) 8天前关闭。 当我尝试
我有以下代码: #include using namespace std; int main() { set S; S.insert("item1"); S.insert("i
在'helper'函数中输出集合元素时,预计元素'2'已经被移除。但实际结果仍然是'1 2 3 4 5'。 但是,集合的大小是 4。 我想知道潜在的问题。 #include #include us
有点难以解释,但我想制作一个以图片为背景的网站,并在图片上覆盖一层灰色。然后我想在覆盖层上方添加一些文本,在文本所在的位置,灰色覆盖层被移除,因此文本是在没有灰色覆盖层的情况下写入的。 我在一年前用图
这是导致错误的代码: 工厂.h: #include #include namespace BaseSubsystems { template class CFactory
我正在研究结构 vector 。 当我试图用迭代器调用这个函数时,像这样: vec2.erase (vec2.begin()+iter2); 它向我发送了这个错误: "no match for 'op
进程文件: eraser or eraser.exe 进程名称: Eraser 进程类别:存在安全风险的进程 英文描述: eraser.exe is the main executable
我正在编写一个简单的程序,它使用了 std::map::erase . 该程序很好,但有一些我不明白的地方。 如果我传给 erase function 第一个迭代器超出第二个迭代器的间隔,该函数不会删
如果我的变量是 0 到 6294 之间的随机数,我如何让我的输出仅打印出第一个变量? 我的随机数代码是 int random = (int)(Math.random() * 6294); 例如,如果它
我正在尝试通过一个测试程序,并且我通过了所有测试,除了涉及到我的删除功能时,程序崩溃了。 我最好的猜测可能是前面或后面的哨兵节点正在被删除。或内存泄漏。 最佳答案 我怀疑有人愿意阅读所有这些代码并为您
背景:这是一个程序,用于将存储在 vector 中的数字逐位加 1。 前面有 0 的数字可以作为输入,但不能作为输出。例如:0123 和 123 都是有效输入 但 0124 是无效输出 而 124 是
我正在尝试创建一个垂直滚动的射击游戏,当您按下空格键时,会创建一颗子弹,然后当子弹离开屏幕时,子弹就会被销毁。我通过声明为 vector bullets; 的 vector 跟踪子弹当我试图销毁屏幕外
我在下面写这段代码,发现了这个奇怪的行为: #include #include #include using namespace std; int main() { map map1;
我为 SFML 编写了一个线程渲染器,它接受指向可绘制对象的指针并将它们存储在一个 vector 中以在每一帧中绘制。开始向 vector 添加对象和从 vector 中删除对象会经常导致段错误 (S
如果我想从 map 中删除单个元素(除了可能的错误检查之外我不关心返回值),有两种方法可以实现:erase按值键或删除: http://ideone.com/YWocN7 #include #inc
我是一名优秀的程序员,十分优秀!