Is an object guaranteed to be moved when it is returned?

前端 未结 3 373
伪装坚强ぢ
伪装坚强ぢ 2020-12-05 15:21

I know that when passing an object by value to a function, the move constructor is always called if there is one, assuming no copy elision. What about returning an object by

相关标签:
3条回答
  • 2020-12-05 15:59

    The rule is that whenever copy elision is allowed but does not occur, the move constructor will be used if it is available, and otherwise the copy constructor will be used.

    The exact behaviour is defined by [class.copy]/32:

    When the criteria for elision of a copy operation are met or would be met save for the fact that the source object is a function parameter, and the object to be copied is designated by an lvalue, overload resolution to select the constructor for the copy is first performed as if the object were designated by an rvalue. If overload resolution fails, or if the type of the first parameter of the selected constructor is not an rvalue reference to the object’s type (possibly cv-qualified), overload resolution is performed again, considering the object as an lvalue.

    0 讨论(0)
  • 2020-12-05 16:12

    In this case, since the return value has a name (f), it would be NRVO (named return value optimization) that would apply.

    So, the technical answer based only on wording, is that the absence of RVO won't prevent copy elision, since NRVO could still allow it.

    Past that, I believe the selection between move/copy of the return value can/will depend on the definition of Foo -- there are definitely times it'll be copied instead of moved, such as if you've explicitly deleted the move constructor and move assignment operators, or you haven't defined move construction/assignment, and it isn't eligible for them being synthesized implicitly.

    Edit: [responding to edited question]: Having a move constructor still doesn't guarantee that the result will be moved. One obvious example would be if you had deleted the move assignment operator, and were assigning the result (rather than using it to initialize). In this case, the deleted move assignment operator would prevent moving the return value.

    To answer what you may have been getting at, though, the general rule is that moving will be done if possible, and it'll fall back to copying if and only if something prevents the result from being moved.

    0 讨论(0)
  • 2020-12-05 16:18

    Yes. See [class.copy] p32

    When the criteria for elision of a copy operation are met or would be met save for the fact that the source object is a function parameter, and the object to be copied is designated by an lvalue, overload resolution to select the constructor for the copy is first performed as if the object were designated by an rvalue. If overload resolution fails, or if the type of the first parameter of the selected constructor is not an rvalue reference to the object’s type (possibly cv-qualified), overload resolution is performed again, considering the object as an lvalue. [ Note: This two-stage overload resolution must be performed regardless of whether copy elision will occur. It determines the constructor to be called if elision is not performed, and the selected constructor must be accessible even if the call is elided. — end note ]

    0 讨论(0)
提交回复
热议问题