C++ Class Move Constructor

Introduction

In addition to copying, we can also move the data from one object to the other.

We call it a move semantics.

Move semantics is achieved through a move constructor and move assignment operator.

The object from which the data was moved, is left in some valid but unspecified state.

The move operation is efficient in terms of speed of execution, as we do not have to make copies.

Move constructor accepts something called rvalue reference as an argument.

Every expression can find itself on the left-hand side or the right-hand side of the assignment operator.

The expressions that can be used on the left-hand side are called lvalues, such as variables, function calls, class members, etc.

The expressions that can be used on the right-hand side of an assignment operator are called rvalues, such as literals, and other expressions.

Now the move semantics accepts a reference to that rvalue.

The signature of an rvalue reference type is T&&, with double reference symbols.

So, the signature of a move constructor is:

MyClass (MyClass&& rhs) 

To cast something to an rvalue reference, we use the std::move function.

This function casts the object to an rvalue reference.

It does not move anything.

An example where a move constructor is invoked:

#include <iostream> 

class MyClass { }; 

int main() //w w  w .  j a v a2s. c  o m
{ 
    MyClass o1; 
    MyClass o2 = std::move(o1); 
    std::cout << "Move constructor invoked."; 
    // or MyClass o2{std::move(o1)}; 
} 

In this example, we define an object of type MyClass called o1.

Then we initialize the second object o2 by moving everything from object o1 to o2.

To do that, we need to cast the o2 to rvalue reference with std::move(o1).

This, in turn, invokes the MyClass move constructor for o2.

If a user does not provide a move constructor, the compiler provides an implicitly generated default move constructor.

Let us specify our own, user-defined move constructor:

#include <iostream> 
#include <string> 

class MyClass //w w w  .j a v a  2  s .  co  m
{ 
private: 
    int x; 
    std::string s; 
public: 
    MyClass(int xx, std::string ss) // user provided constructor 
         : x{ xx }, s{ ss } 
    {} 

    MyClass(MyClass&& rhs) // move constructor 
         : 
        x{ std::move(rhs.x) }, s{ std::move(rhs.s) } 
    { 
        std::cout << "Move constructor invoked." << '\n'; 
    } 
}; 

int main() 
{ 
    MyClass o1{ 1, "Some string value" }; 
    MyClass o2 = std::move(o1); 
} 

This example defines a class with two data members and two constructors.

The first constructor is some user-provided constructor used to initialize data members with provided arguments.

The second constructor is a user-defined move constructor accepting an rvalue reference parameter of type MyClass&& called rhs.

This parameter will become our std::move(o1) argument/object.

Then in the constructor initializer list, we also use the std::move function to move the data fields from o1 to o2.




PreviousNext

Related