Understanding Copy Construction in C++: A Detailed Guide with Examples

Gaurav Sah
5 min readMay 20, 2024

--

Copy construction is a fundamental concept in C++, central to the management of object copying. This guide will delve into the nuances of copy construction, provide detailed examples, and explore best practices for using and customizing copy constructors in C++.

What is a Copy Constructor?

A copy constructor is a special constructor in C++ that initializes a new object as a copy of an existing object. The copy constructor has the following signature:

ClassName(const ClassName& other);

Here, ClassName is the name of the class, and other is a reference to an object of the same class.

When is a Copy Constructor Called?

The copy constructor is invoked in several scenarios:

  1. When an object is initialized with another object of the same type.
  2. When an object is passed by value to a function.
  3. When an object is returned by value from a function.
  4. When an object is thrown or caught by value.

Default Copy Constructor

If you do not explicitly define a copy constructor, the compiler provides a default one. The default copy constructor performs a shallow copy of the object, copying all member variables bitwise from the source object to the target object. This works fine for classes that do not manage resources like dynamic memory, file handles, or network connections.

Example:

#include <iostream>

class Simple {
public:
int data;
};

int main() {
Simple obj1;
obj1.data = 10;

Simple obj2 = obj1; // Default copy constructor is called

std::cout << "obj2.data = " << obj2.data << std::endl; // Outputs 10

return 0;
}

User-Defined Copy Constructor

For classes that manage resources, a user-defined copy constructor is essential to ensure proper copying of resources. Let’s illustrate this with an example.

Example:

#include <iostream>

class DeepCopy {
private:
int* data;
public:
// Constructor
DeepCopy(int value) {
data = new int;
*data = value;
}

// Copy Constructor
DeepCopy(const DeepCopy& other) {
data = new int; // Allocate new memory
*data = *(other.data); // Copy the value
}

// Destructor
~DeepCopy() {
delete data;
}

// Display data
void display() const {
std::cout << "Value: " << *data << std::endl;
}
};

int main() {
DeepCopy obj1(42);
DeepCopy obj2 = obj1; // User-defined copy constructor is called

obj1.display(); // Outputs: Value: 42
obj2.display(); // Outputs: Value: 42

return 0;
}

In this example, the DeepCopy class manages a dynamically allocated integer. The user-defined copy constructor ensures that obj2 gets its own copy of the dynamically allocated integer rather than sharing it with obj1.

Copy Constructor and Const-Correctness

The parameter of the copy constructor is a constant reference (const ClassName& other). This ensures that the source object is not modified during the copying process.

Example:

#include <iostream>

class Sample {
public:
int data;

// Copy Constructor
Sample(const Sample& other) {
data = other.data; // OK, because other is const
}
};

int main() {
Sample obj1;
obj1.data = 20;

Sample obj2 = obj1; // Copy constructor is called

std::cout << "obj2.data = " << obj2.data << std::endl; // Outputs 20

return 0;
}

Copy Constructor in Inheritance

When dealing with inheritance, the copy constructor of the base class is called automatically when copying derived class objects. However, you might need to explicitly define a copy constructor in the derived class to ensure proper copying of derived class members.

Example:

#include <iostream>

class Base {
public:
int base_data;

Base(int value) : base_data(value) {}

// Base class copy constructor
Base(const Base& other) {
base_data = other.base_data;
}
};

class Derived : public Base {
public:
int derived_data;

Derived(int base_value, int derived_value) : Base(base_value), derived_data(derived_value) {}

// Derived class copy constructor
Derived(const Derived& other) : Base(other) { // Call base class copy constructor
derived_data = other.derived_data;
}
};

int main() {
Derived obj1(10, 20);
Derived obj2 = obj1; // Derived class copy constructor is called

std::cout << "obj2.base_data = " << obj2.base_data << std::endl; // Outputs 10
std::cout << "obj2.derived_data = " << obj2.derived_data << std::endl; // Outputs 20

return 0;
}

In this example, the Derived class inherits from the Base class. The copy constructor of Derived calls the copy constructor of Base to ensure that the base part of the Derived object is correctly copied.

Copy Elision

Modern C++ compilers perform an optimization called copy elision, which can omit the call to the copy constructor in certain situations, such as when returning an object by value. This optimization can significantly improve performance by eliminating unnecessary copies.

Example:

#include <iostream>

class Example {
public:
int data;

Example(int value) : data(value) {}

Example(const Example& other) {
data = other.data;
std::cout << "Copy constructor called" << std::endl;
}
};

Example createExample() {
return Example(30); // Copy elision might occur
}

int main() {
Example ex = createExample(); // Copy constructor might not be called
std::cout << "ex.data = " << ex.data << std::endl; // Outputs 30

return 0;
}

In this example, the compiler might optimize away the call to the copy constructor when createExample returns an Example object.

Rule of Three, Five, and Zero

When you need to define a copy constructor, you often also need to define a destructor and a copy assignment operator. This is known as the Rule of Three. With C++11 and later, the Rule of Three has expanded to the Rule of Five, which includes the move constructor and move assignment operator.

Rule of Three Example:

#include <iostream>

class Resource {
private:
int* data;
public:
// Constructor
Resource(int value) {
data = new int(value);
}

// Copy Constructor
Resource(const Resource& other) {
data = new int(*(other.data));
}

// Copy Assignment Operator
Resource& operator=(const Resource& other) {
if (this == &other) {
return *this; // Handle self-assignment
}
delete data; // Free existing resource
data = new int(*(other.data)); // Copy the resource
return *this;
}

// Destructor
~Resource() {
delete data;
}

// Display data
void display() const {
std::cout << "Value: " << *data << std::endl;
}
};

int main() {
Resource res1(100);
Resource res2 = res1; // Copy constructor
Resource res3(200);
res3 = res1; // Copy assignment operator

res1.display(); // Outputs: Value: 100
res2.display(); // Outputs: Value: 100
res3.display(); // Outputs: Value: 100

return 0;
}

Conclusion:

Copy constructors are crucial for managing resources and ensuring correct object behaviour in C++. Understanding when and how to use them, and following best practices like the Rule of Three, can help you write robust and efficient C++ code. Whether dealing with simple classes or complex inheritance hierarchies, mastering copy construction is essential for any serious C++ programmer.

--

--

Gaurav Sah
Gaurav Sah

Written by Gaurav Sah

Full Stack Developer and entrepreneur specializing in MERN stack. I build web apps and write about tech, trends, and startups. Let's create something amazing!

No responses yet