Move construction from !is(T == typeof(this))
Stanislav Blinov via Digitalmars-d
digitalmars-d at puremagic.com
Mon Apr 24 07:27:00 PDT 2017
On Monday, 24 April 2017 at 04:21:36 UTC, Manu wrote:
> struct X {}
>
> struct Y {
> this(auto ref X x)
> {
> static if (__traits(isRef, x))
> {
> // x is lvalue, copy construct
> }
> else
> {
> // x MUST be rvalue(?), move construct
> // does this pattern require that I invalidate x the same
> way C++ does such that X's destructor won't clean up or crash?
"Require" is a bit too strict. But yes, if the pattern is to
assume ownership of x's contents, then you should clear it.
> Is this solid? I have a vague memory of thinking on this once
> and realising there was some edge case where it was logically
> flawed, but I can't remember clearly.
>
> Assuming this does work, the next issue is something that
> mirrors std::move(), is there already such a thing?
We can't truly override move semantics in D. The language went
"all in" and left little leeway to us users in that regard.
Interestingly enough, we *can* come close in cases such as the
code below.
> struct X {}
>
> struct Y {
> this(ref const X x)
> {
> // x is lvalue reference, copy construct
> }
> this(X x)
> {
> // x is an lvalue... which may be a copy of another lvalue.
> can't move
> construct :/
Why not? The first overload will be called with an lvalue, i.e:
X x;
Y y1 = x;
The second one will be called in all other cases:
Y y2 = X();
Y y3 = move(x); // move(x) returns X
For second overload, x will be destructed once the ctor returns.
I don't see any reason why you wouldn't move it's guts out.
> I guess the question in this case is how overload selection may
> or may not work...
> I didn't test this, but I expect it's an ambiguous call given
> an lvalue?
There is no ambiguity there as far as I can tell.
Same goes for opAssign, you can have
ref opAssign(ref X x) { /*...*/ } // assign lvalue
ref opAssign(X x) { /*...*/ } assign rvalue
I guess you mean that you don't have source for X. But in that
case, you won't be able to "move-construct" Y from X in C++
either. Nor would the compiler. If you at least know the exact
memory layout of X, you could hack together a custom move:
this(X x)
{
auto px = cast(XLayout*)cast(void*)&x;
// move from px...
}
...but that's neither here nor there. The words "dangerous",
"non-portable" and UB march with pitchforks right behind that
pattern ;)
More information about the Digitalmars-d
mailing list