Skip to main content

Destructor

Similar to constructor, a class has another special member function called destructor. Destructor cleans up the object.  

When an object goes out of scope, the destructor is automatically called. Destructor is also called when the pointer to the object is released using delete operator.

Destructor has a same name as class and is preceded by tilde (~) symbol. 

Class A has a destructor ~A()
 
Destructor should be used to clean up the object -  release memory and other resources, close files, stop threads etc.

#include <iostream>
using namespace std; 
class Arr
{
 int *elements;
 int len;
public:
 Arr(int len=5);
 ~Arr();//destructor
 int & operator[](int index);
};
Arr::Arr(int len)
{
 elements = new int[len];
}
Arr::~Arr()
{
 delete []elements;
}
int & Arr::operator[](int index)
{
 return elements[index];
} 
int main()
{
 Arr obj(10);
 for(int i=0;i<10;i++)
 obj[i] = i*i; 
} 
 

In the above code, you see the destructor of class Arr, which  releases the memory  allocated by constructor. The object obj is destroyed when main function terminates, and its destructor is called which releases memory.

Destructor takes no parameters. So destructor can not be overloaded.

If there is no user defined destructor, then compiler provides a trivial destructor.

The example shown above uses dynamic memory and hence needs user defined destructor to release memory.

Order of destructor call


Destructor of an object is called when the object is being destroyed - that is when the object goes out of scope.
  1. For global objects, destructor is called at the end of the program.
  2. For local objects, it is called when the block exits.
  3. For parameters and return values, it is called when the function exits.
  4. For dynamically created objects, destructor is called when delete operator is used.
Also note that the order of destructors is the opposite of order of constructors. Which means that the object created last, is destroyed first.

The program below shows the order of destructors.

class A
{
 int num;
public:
 A(int n):num(n){cout<<"constructor"<<n<<endl;}
 ~A()
 {
 cout<<"Destructor";
 }
};
int main()
{
 A obj1(10); 
 if(10>5)
 {
 A obj3(15);
 }
 A obj2(12);
}

The output of the program above will be
constructor10
constructor15
Destructor15
constructor12
Destructor12
Destructor10

Note :
  • Compiler provides the following 4 functions automatically for a class
    1. Default constructor - if the class has no other constructors
    2. Copy Constructor
    3. Destructor
    4. Assignment operator
  • Destructor should be public. If not, objects of class can not be destroyed.
You will find all these notes, quiz and complete programs in Simplified C++ app by Hegdeapps 

Comments

Popular posts from this blog

Polymorphism

You hear the term Polymorphism too frequently with object oriented languages. Along with Inheritance and Encapsulation, polymorphism is one of the corner stones of object oriented design. What is Polymorphism, exactly? Polymorphism is a mechanism by which you provide single interface for multiple methods. (poly - many, morph - form). In C++, polymorphism can be compile time or run-time. Compile time polymorphism is provided with the help of overloaded operators/functions and templates. Run time polymorphism is provided with the help of virtual functions and late binding. Late Binding: Connecting a function call to function body is called binding. Most functions use early binding where this binding happens before the program is run - during compile time. This is also called static binding. Late binding (also called dynamic binding)  is when a function call is connected to function body at run time. This is done after looking at the type of the object. Late binding is ach

Operator Overloading

What is operator overloading ? Operator overloading is the process of customizing C++ operators for operands of user defined types.   When you have two objects of a class- num1 and num2 , you can write a function to add them such as  ans = add(num1,num2); That does not look neither simple nor intuitive. You would prefer to write      ans = num1+num2; as you would write expressions for basic data types like integers, floats etc.    This can be done using Operator overloading. Operator overloading lets you write such statements. That is, it lets you call your functions on objects using  +, - ,* etc.    + operator will call addition function on the object (when you write op. overloading function for +). * will call multiply on objects etc. Names of overloaded operator functions start with keyword operator followed by  symbol of the operator. e.g. +, - etc. Unary operator functions take 0 parameters for members. The operand for these function is the object cal

Constant members of a Class

A constant  is a value which can not be modified. As in C, we can have literal constants using #define and we can have enums and we can define a variable to be const .  Let us look at const variables here. By declaring a variable as const , we ensure that it is not modified accidentally. Any modification to a constant will give a compilation error. A const should always be initialized while defining. In the program below,  assignment to pi gives a compiler error because pi is defined as const and code is trying to modify this. int main () { const float pi = 22.0/7 ; int radius = 12 ; radius ++ ; /*ok*/ pi = 3.14 ; /*error*/ }   Constant parameters to functions Even function parameters can be const ant. We have earlier discussed that making a reference parameter as constant will avoid the function from accidentally modifying the argument. void printnum ( int & n) { cout << n ++ ; } void printnum2 ( const int & n) { cout <&l