移动和擦除后对象的容器字段为空

Object's container field is empty after move and erase

我有两个向量,我想将一个对象从一个向量移动到另一个向量,在移动它之前,我创建了一个指向该对象的指针。但是当我擦除第一个向量中的索引时,如果我通过指针访问它,则对象内的容器为空。很难描述,一些代码会更好地解释它:

Word.h

class Word
{
public:
    Word(string text);
    ~Word();

    string text;
};

Card.h

class Card
{
public: 
    Card(vector<Word>& words);
    ~Card();

    vector<Word> words;
};

主要()

vector<Word> words {
    Word("dog"),
    Word("cat"),
    Word("horse")
};

Card card(words);

vector<Card> deck1;
vector<Card> deck2;

deck1.push_back(move(card));

Card* p = &deck1[0];
deck2.push_back(move(deck1[0]));
cout << p->words.size() << endl; // 3

deck1.erase(deck1.begin());

cout << p->words.size() << endl; // 0 <-- Why is size() 0 here?

p = &deck2[0];
cout << deck2[0].words.size() << endl; // 3 <-- and 3 here
cout << p->words.size() << endl; // <-- and 3 here! 

getchar();

这些移动没有效果,因为 Card 没有移动构造函数或移动赋值运算符。向量 deck1deck2 包含推入其中的任何内容的副本。

但问题(或者至少是一个问题)是当您从 deck1 中删除时您使 p 无效。取消引用 p 是未定义的行为 (UB)。

deck1.push_back(move(card));
Card* p = &deck1[0];
....
deck1.erase(deck1.begin()); // p invalidated
....
cout << p->words.size() << endl; // UB, anything can happen

p = &deck2[0];
cout << deck2[0].words.size() << endl;
cout << p->words.size() << endl; // No problem, p points to deck2[0]