C++, std::move(), POD типы и неопределенное поведение

347
03 ноября 2021, 06:40

У меня возник вопрос по семантике перемещения в рамках POD типов. Он сложный, поэтому я постараюсь разбить его на подпункты:

  1. В каком состоянии (согласно Стандарту) находится объект после перемещения? Корректное, но неопределенное? Что это значит?

  2. Если объект является объектом POD типа, то каков результат операции std::move(object)?

  3. Относятся ли элементарные типы (int, float, double) к POD типам?

  4. Если переместить объект POD типа, то законно ли дальнейшее использование такого объекта? Например:

    struct Value
    {
        int i;
    };
    Value value;
    value.i = 1;
    do_something(std::move(value));
    // Дальнейшая работа с value.
    

Буду очень благодарен, если кто-нибудь поможет мне разобраться в этих вопросах.

Answer 1
  1. Не "корректное, но неопределенное", а "корректное, но неспефицированное" (!) (http://eel.is/c++draft/definitions#defns.valid).

    Термин "неопределенный" обладает в языке С++ негативной коннотацией (см. "неопределенное поведение"). У стандарта языка С++ нет никаких причин привносить такой оттенок в описание состояния перемещенного объекта. Поэтому именно "неспефицированное", а не "неопределенное".

    Упоминание неспецифицированного состояния означает, что все типы разные, и стандарт языка не может однозначно специфицировать конкретные единые правила для всех типов. Каждый тип в рамках семантики перемещения может вести себя по-своему. Язык говорит вам, что за деталями поведения каждого конкретного типа надо обращаться в документацию этого конкретного типа. Документация конкретного типа может частично специфицировать состояние объекта после перемещения, а может и специфицировать его полностью. Более того, даже в рамках конкретного типа состояние объекта после перемещения может зависеть от его состояния до перемещения, то есть от факторов времени выполнения.

    Не забывайте при этом, что "перемещение", по определению - более широкая концепция, чем "копирование". Обыкновенное "копирование" является, по определению, частным случаем "перемещения". То есть не будет ничего удивительного в том, если вы, обратившись к спецификации какого-либо типа, обнаружите, что для этого типа "перемещение" эквивалентно "копированию". Если, обратившись к спецификации некоего конкретного типа, вы увидите, что для данного типа семантика перемещения эквивалентна семантике копирования, что означает, что исходный объект после перемещения полностью сохраняет свое исходное корректное состояние.

    Также стоит заметить, что термин "неспефицированное поведение" исторически тоже наделялся явным негативным оттенком: не стоит завязываться в своем коде на такое неспефицированное поведение, как, например, порядок вычисления аргументов функции. Однако в случае семантики перемещения ситуация несколько иная: завязываться на документированные свойства перемещенных объектов вполне можно (и нужно), пока вы не выходите за рамки того, что действительно специфицировано.

  2. Для всех типов результатом операции std::move является xvalue того типа, который был передан в качестве аргумента. std::move - это лишь преобразование типов. std::move сам по себе больше ничего не делает. Поэтому никаких особенностей, специфичных для POD, типов у std::move нет.

  3. Да, относятся.

  4. Да, законно. Для POD типов операция перемещения определена как обыкновенное копирование (см. пункт 1).

Answer 2
  1. Если это объект класса, то его состояние после перемещения зависит только от того, что написано в перемещающем конструкторе / операторе присваивания.

    Если это не объект класса (а, например, скаляр), то перемещение делает то же, что и копирование. Исходный объект не меняется.

    В "корректное, но неопределенное состояние" попадают некоторые стандартные классы после перемещения. Например std::string: "left in valid, but unspecified state". (А некоторые нет. Например std::vector: "guaranteed to be empty()"`.).

    Корректное, но неопределенное? Что это значит?

    Нуу, это и значит.

    Например, строка может оказаться пустой, а может сохранить свое старое значение - это неопределенное состояние.

    Или например, если .size() возвращает не ноль, то естественно можно спокойно обращаться к строка[0], не опасаясь таинственных вылетов или ошибок - состояние строки корректное.

  2. std::move сам по себе ничего не делает, не важно к POD его применили или нет. Он только превращает выражение в rvalue, чтобы его можно было передать в перемещающий конструктор / оператор присваивания (или еще куда-то, где нужен rvalue).

  3. Относятся ли элементарные типы (int, float, double) к POD типам?

    Да. См. определение POD.

  4. Если переместить объект POD типа, то законно ли дальнейшее использование такого объекта?

    Да, объект не изменится. См. (1).

Answer 3
  1. "Корректное, но неопределенное" Значит закладываться на что либо нельзя, на практике POD просто копируется, другие объекты могут свопаться, кто-то может "обнульть" перемещенный объект. Но после std::move что либо делать с объектом лучше не стоит.
  2. На усмотрение компилятора, но обычно ничего не изменяется, но гарантировать, что с изменение версии компилятора ничего не поменяется никто не будет.
  3. да
  4. Законно, но результат не определен.
READ ALSO
Как сделать относительный путь к библиотеке?

Как сделать относительный путь к библиотеке?

Как сделать относительный путь к dll?

192
C++, свойста шаблонов, работа с памятью

C++, свойста шаблонов, работа с памятью

Подскажите, пожалуйста, почему вышеуказанный код показывает, что

116
Программа не работает при смене функций местами

Программа не работает при смене функций местами

Почему-то программа крашится на scanf'е, выдаёт ошибку malloc_consolidate() invalid chunk sizeНо если я поставлю функцию game_logic () в main после остальных, то всё будет...

258
Задача на проверку последовательности

Задача на проверку последовательности

Я не могу понять что не так с кодом, можете помочь дописать его

103