Lesson 2
Destructors in C++
Lesson Introduction

Hello! Today, we are going to learn about a special type of function in C++ called a destructor. Destructors play a crucial role in managing resources, like memory, in programming. If you've ever played with toys and had to clean up, think of a destructor as the tool that helps clean up after the objects your program uses. By the end of this lesson, you'll understand what destructors are, how to use them, and why they're important.

Understanding Destructors

In C++, a destructor is a special member function of a class that runs when an object of that class goes out of scope or is explicitly deleted. Imagine you have a paper airplane, and when you're done playing with it, you need to recycle the paper. A destructor helps with that recycling process in programming.

Destructors are vital because they ensure that resources, like memory or files, are properly released. This prevents memory leaks.

Destructor Syntax

A destructor has the same name as the class but is preceded by a tilde (~). It doesn't take any arguments and doesn't return any value. Additionally, a class can only have one destructor, you can't overload it.

Here's a simple example:

C++
1class MyClass { 2public: 3 // Constructor 4 MyClass() { 5 // Constructor code 6 } 7 8 // Destructor 9 ~MyClass() { 10 // Destructor code 11 } 12};

This class MyClass has a constructor (to set up the object) and a destructor (to clean up the object). If you don't define a destructor, the compiler provides a default one. However, it's a good practice always to declare your own destructor if your class manages resources like dynamic memory, file handles, or network connections. This ensures that resources are correctly and explicitly released.

You can also define the destructor outside the class, just like with other methods:

C++
1// Destructor implementation outside the class 2MyClass::~MyClass() { 3 // Destructor code 4}
Destructors in Action

Let's explore how destructors work with a practical example. We will create a class called MyClass, and our program will print messages when the constructor and destructor are called.

C++
1#include <iostream> 2 3class MyClass { 4public: 5 MyClass() : data(nullptr) { 6 std::cout << "Constructor called!" << std::endl; 7 } 8 ~MyClass() { 9 std::cout << "Destructor called!" << std::endl; 10 } 11 12private: 13 int* data; 14}; 15 16int main() { 17 MyClass obj; // Constructor called here 18 return 0; // Destructor called here 19}

Here's what's happening:

  • When obj is created, the constructor runs and prints "Constructor called!".
  • When obj goes out of scope (at the end of main), the destructor runs and prints "Destructor called!".
Key Points to Remember
  • Automatic Calling: Destructors are called automatically when an object goes out of scope or is deleted.
  • Only One Destructor: A class can have only one destructor.
  • Default Destructor: If you don't define a destructor, the compiler provides a default one.
  • Write Outside the Class: You can write the body of the destructor outside of the class definition.
  • No Throwing Exceptions: A destructor should not throw exceptions because it might not be safely caught if an exception is thrown during the destruction process. This is because destructors can be invoked during the stack unwinding process of another exception, leading to a call to std::terminate.
Practical Use Cases

Now, let’s look at an example where a destructor is very useful: managing dynamic memory. Suppose we have a MyClass that allocates an array dynamically. The destructor will handle deleting this array.

C++
1#include <iostream> 2 3class MyClass { 4private: 5 int* data; 6public: 7 MyClass(int size) : data(new int[size]) { 8 std::cout << "Memory allocated!" << std::endl; 9 } 10 ~MyClass() { 11 delete[] data; // Deallocate memory 12 std::cout << "Memory deallocated!" << std::endl; 13 } 14}; 15 16int main() { 17 MyClass obj(10); // Memory allocated here 18 // Memory deallocated here 19}

This example shows how destructors help manage dynamic memory effectively. In real life, just as you might need to dispose of garbage properly to keep your room clean, using destructors ensures your program doesn't waste memory or other resources.

Lesson Summary

Today, we learned that destructors are special functions that clean up when objects are no longer needed. They:

  • Automatically run when an object goes out of scope or is deleted.
  • A class can have only one destructor.
  • Are critical for releasing resources and preventing memory leaks.
  • Should not throw exceptions to ensure safe execution.

Destructors make sure your programs are tidy and efficient, just like cleaning up after playing keeps your room neat.

Great job making it through the lesson! Now it's time to practice. You'll apply what you've learned about destructors by writing and running some code to see how they work in real scenarios. Let's dive into the hands-on activities!

Enjoy this lesson? Now it's time to practice with Cosmo!
Practice is how you turn knowledge into actual skills.