Editor's note: this question was asked before Rust 1.0 and some of the assertions in the question are not necessarily true in Rust 1.0. Some answers have been updated to address both versions.
I have this struct
struct Triplet {
one: i32,
two: i32,
three: i32,
}
If I pass this to a function, it is implicitly copied. Now, sometimes I read that some values are not copyable and therefore have to moved.
Would it be possible to make this struct Triplet
non-copyable? For example, would it be possible to implement a trait which would make Triplet
non-copyable and therefore "movable"?
I read somewhere that one has to implement the Clone
trait to copy things that are not implicitly copyable, but I never read about the other way around, that is having something that is implicitly copyable and making it non-copyable so that it moves instead.
Does that even make any sense?
Preface: This answer was written before opt-in built-in traits—specifically the
Copy
aspects—were implemented. I've used block quotes to indicate the sections that only applied to the old scheme (the one that applied when the question was asked).Types now move by default, that is, when you define a new type it doesn't implement
Copy
unless you explicitly implement it for your type:The implementation can only exist if every type contained in the new
struct
orenum
is itselfCopy
. If not, the compiler will print an error message. It can also only exist if the type doesn't have aDrop
implementation.To answer the question you didn't ask... "what's up with moves and copy?":
Firstly I'll define two different "copies":
(&usize, u64)
, it is 16 bytes on a 64-bit computer, and a shallow copy would be taking those 16 bytes and replicating their value in some other 16-byte chunk of memory, without touching theusize
at the other end of the&
. That is, it's equivalent to callingmemcpy
.Rc<T>
involves just increasing the reference count, and a semantic copy of aVec<T>
involves creating a new allocation, and then semantically copying each stored element from the old to the new. These can be deep copies (e.g.Vec<T>
) or shallow (e.g.Rc<T>
doesn't touch the storedT
),Clone
is loosely defined as the smallest amount of work required to semantically copy a value of typeT
from inside a&T
toT
.Rust is like C, every by-value use of a value is a byte copy:
They are byte copies whether or not
T
moves or is "implicitly copyable". (To be clear, they aren't necessarily literally byte-by-byte copies at run-time: the compiler is free to optimise the copies out if code's behaviour is preserved.)However, there's a fundamental problem with byte copies: you end up with duplicated values in memory, which can be very bad if they have destructors, e.g.
If
w
was just a plain byte copy ofv
then there would be two vectors pointing at the same allocation, both with destructors that free it... causing a double free, which is a problem. NB. This would be perfectly fine, if we did a semantic copy ofv
intow
, since thenw
would be its own independentVec<u8>
and destructors wouldn't be trampling on each other.There's a few possible fixes here:
w
has its own allocation, like C++ with its copy constructors.v
can no longer be used and doesn't have its destructor run.The last is what Rust does: a move is just a by-value use where the source is statically invalidated, so the compiler prevents further use of the now-invalid memory.
Types that have destructors must move when used by-value (aka when byte copied), since they have management/ownership of some resource (e.g. a memory allocation, or a file handle) and its very unlikely that a byte copy will correctly duplicate this ownership.
"Well... what's an implicit copy?"
Think about a primitive type like
u8
: a byte copy is simple, just copy the single byte, and a semantic copy is just as simple, copy the single byte. In particular, a byte copy is a semantic copy... Rust even has a built-in traitCopy
that captures which types have identical semantic and byte copies.Hence, for these
Copy
types by-value uses are automatically semantic copies too, and so it's perfectly safe to continue using the source.As mentioned above, opt-in built-in traits are implemented, so the compiler no longer has automatic behaviour. However, the rule used for the automatic behaviour in the past are the same rules for checking whether it is legal to implement
Copy
.The easiest way is to embed something in your type that is not copyable.
The standard library provides a "marker type" for exactly this use case: NoCopy. For example: