Program memory: Heap vs Stack
Dynamic memory is allocated on the "heap". The heap is the region of computer memory which is managed by the programmer using pointers to access the memory.
Heap memory details:
- can be accessed globally
- your limit is the physical memory limit of your system (real and virtual)
- user managed memory. Programmer must allocate and free memory.
- memory access is via pointer and thus requires dereferencing to access, requiring extra operations and time
- memory can become defragmented as memory is allocated and freed
- memory can be resized (eg using realloc())
The "stack" is the region of memory which stores declared variables in main() and all called functions. When a function returns, the memory space for the declared variables is released ("popped").
The stack is FILO (first in - lst out) and unwinds variables in memory as functions return.
- the "stack" grows and shrinks as functions push and pop local variables
- the "stack" is managed by the OS
- the default "stack" size is limited by the OS. A stack overflow occurs when this limit is exceeded (eg endless function recursion).
Example of behavior of objects declared on the heap vs stack:
17 | cout << "Constructor called. Value of ii=" << ii << endl; |
23 | cout << "Constructor called. Value of ii=" << ii << endl; |
28 | cout << "Destructor called. Value of ii=" << ii << endl; |
33 | cout << "Start of program" << endl; |
39 | cout << "End of program" << endl; |
Compile:
g++ test.cpp
Run:
./a.out
Start of program
Constructor called. Value of ii=1
Constructor called. Value of ii=2
Constructor called. Value of ii=3
Constructor called. Value of ii=4
Destructor called. Value of ii=3
End of program
Destructor called. Value of ii=2
Destructor called. Value of ii=1
Note:
- Stack: The objects a1 and a2 are declared on the stack.
The constructors are called automatically where they are declared.
The destructors are called automatically when they go out of scope (as defined by the last "}").
- Heap: Objects allocated dynamically on the heap with "new" automatically have their constructors called.
Note that the destructor for a3 is called when the memory is freed with "delete". Delete is not called for a4 and thus the destructor is not called.
C malloc and free vs C++ new and delete:
C malloc() and free() do not call constructors or destructors.
C++ new and delete operators are "class aware" and will call class constructors when a class is allocated with new and a destructor when delete is called. [example]
Mixing malloc with delete or new with free is undefined.
In addition, C++ new and delete
operators were introduced with the C++ practice of allocating memory in
constructors and deallocating the same memory with a destructor.
new/delete advantages:
- new/delete invokes constructor/destructor. Malloc/free will not.
- new does not need typcasting. Malloc requires typcasting the returned pointer.
- new/delete operators can be overloaded, malloc/free can not.
- new does not require you to explicitly calculate the quantity of memory required. (Unlike malloc)
C dynamic memory allocation:
Use "malloc", "calloc" and "free": File: MallocTest.cpp
17 | char *str1 = ( char *) "This is a text string" ; |
18 | char *str2 = ( char *) malloc ( strlen (str1)); |
21 | SSS *s1 = (SSS *) calloc (4, sizeof (SSS)); |
24 | if ( errno == ENOMEM) printf ( "Error ENOMEM: Insufficient memory available\n" ); |
29 | for (kk=0; kk < 5; kk++) |
34 | for (jj=0; jj < 5; jj++) |
36 | printf ( "Value stored: %d\n" ,s1[jj].ii); |
Compile: gcc -o MallocTest MallocTest.cpp
Run: MallocTest
Value stored: 0
Value stored: 1
Value stored: 2
Value stored: 3
Value stored: 4
Manual pages:
- malloc: a memory allocator
- calloc: a memory allocator
- realloc: memory reallocator
- free: frees the memory space pointed to by ptr
- errno: obtains error number of last error
- perror: obtains error number of last error
Note that calloc() zeros memory while malloc() does not.
C++ dynamic memory allocation:
Use "new" and "delete": File: AllocNewTest.cpp
18 | CCC::CCC( int _ii, double _dd) |
27 | CCC *cc1 = new CCC(4, 5.5); |
28 | CCC *cc2 = new CCC[5]; |
30 | CCC **c4 = new CCC * [5]; |
38 | cout << cc1->ii << endl; |
39 | cout << cc2[3].ii << endl; |
40 | cout << cc3.ii << endl; |
41 | cout << c4[0]->ii << endl; |
42 | cout << c4[1]->ii << endl; |
Note the difference between:
- new CCC(3) creates memory for a single object who's integer member is set to 3.
- new CCC[3] creates memory for three objects of type CCC and no variables are set.
Compile: g++ -o AllocNewTest AllocNewTest.cpp
Run:
./a.out
5
6
7
8
9
C function returning a pointer vs C++ function returning a copy:
C function returning a pointer:
04 | char *fnConvert( int _ii) |
06 | char *str = ( char *) malloc (10); |
09 | fprintf (stderr, "Error: Memory allocation failed.\n" ); |
12 | sprintf (str, "%d" , _ii); |
18 | char *s1=fnConvert( 34567 ); |
C++ function returning a copy:
07 | string fnConvert( int _ii) |
16 | cout << fnConvert( 34567 ) << endl; |
Note: The STL string class copy constructor is employed to return a
copy (return by value). Yes the variable "ost" is out of scope once we
leave the function, but the copy of its contents is valid.
Do not "return ost.str().c_str()" as this pointer is out of scope once
the procesing leaves the function and the data lost.
C++ dynamic memory allocation exception handling:
Use exception handling:
12 | for ( ii=0; ii < 5000000; ii++) |
14 | ptr[ii] = new double [5000000]; |
18 | catch ( bad_alloc &memmoryAllocationException ) |
20 | cout << "Error on loop number: " << ii << endl; |
21 | cout << "Memory allocation exception occurred: " |
22 | << memmoryAllocationException.what() |
27 | cout << "Unrecognized exception" << endl; |
Compile: g++ -o AllocNewTest AllocNewTest.cpp
Run:
- Observer system limits: ulimit -a
- Set system limits: ulimit -m 100
- Run with fewer privileges: nice -n 19 AllocNewTest
C++ Virtual Destructors:
Using polymorphism to delete dynamically allocated objects by
their base class pointer: Declare base class destructor virtual so that
the delete operator can be applied to the base class pointer. The
derived class destructor is called first, before the base class
destructor.
08 | virtual ~Base(){ cout << "Base class destructor called" << endl; } |
11 | class Derived : public Base |
15 | ~Derived(){ cout << "Derived class destructor called" << endl; } |
21 | Base *ptr = new Derived(); |
Resutls:
Derived class destructor called
Base class destructor called
Note:
- If the delete operator is applied to the base class and the
destructor is NOT virtual, then this will cause a memory leak as only a
portion of the memory is freed.
- Base class destructor is not pure virtual (set =0) or there would be no base class implementation of the destructor..
- Class contructors can NOT be virtual.
Notes and Tutorials:

Books: