Memory Management

Section 6: Memory Management

Lesson 1: Dynamic Memory Allocation in C

1.1 malloc, free, calloc, realloc Functions

Dynamic memory allocation in C involves using functions like malloc, free, calloc, and realloc to manage memory on the heap.

Example (Dynamic Memory Allocation in C): 

#include <stdio.h>

#include <stdlib.h>


int main() {

    // Allocating memory for an integer

    int* numPtr = (int*)malloc(sizeof(int));

    if (numPtr != NULL) {

        *numPtr = 42;

        printf("Dynamic Integer: %d\n", *numPtr);


        // Freeing allocated memory

        free(numPtr);

    }


    // Allocating memory for an array of integers

    int* arrPtr = (int*)calloc(3, sizeof(int));

    if (arrPtr != NULL) {

        arrPtr[0] = 1;

        arrPtr[1] = 2;

        arrPtr[2] = 3;


        // Reallocating memory for an expanded array

        arrPtr = (int*)realloc(arrPtr, 5 * sizeof(int));


        // Freeing allocated memory

        free(arrPtr);

    }


    return 0;

}


1.2 Memory Leaks and Memory Management Practices

Memory leaks occur when allocated memory is not properly deallocated. Adopting best practices, such as freeing memory after use and checking for null pointers, helps prevent memory leaks.

Example (Memory Management Practices in C): 

#include <stdlib.h>


int main() {

    // Allocating memory for a string

    char* str = (char*)malloc(10 * sizeof(char));

    if (str != NULL) {

        // Use the allocated memory


        // Freeing allocated memory

        free(str);

    }


    // Check for null pointer after freeing

    if (str != NULL) {

        // Attempting to access str after freeing may lead to issues

    }


    return 0;

}


Lesson 2: Memory Management in C++

2.1 Introduction to new and delete Operators

C++ introduces the new and delete operators for dynamic memory allocation and deallocation, providing a more convenient and type-safe approach.

Example (Dynamic Memory Allocation in C++): 

#include <iostream>


int main() {

    // Allocating memory for an integer

    int* numPtr = new int;

    if (numPtr != nullptr) {

        *numPtr = 42;

        std::cout << "Dynamic Integer: " << *numPtr << std::endl;


        // Deallocating memory

        delete numPtr;

    }


    // Allocating memory for an array of integers

    int* arrPtr = new int[3];

    if (arrPtr != nullptr) {

        arrPtr[0] = 1;

        arrPtr[1] = 2;

        arrPtr[2] = 3;


        // Reallocating memory for an expanded array

        int* newArrPtr = new int[5];

        std::copy(arrPtr, arrPtr + 3, newArrPtr);

        delete[] arrPtr;

        arrPtr = newArrPtr;


        // Deallocating memory

        delete[] arrPtr;

    }


    return 0;

}


2.2 Smart Pointers in Modern C++

Modern C++ introduces smart pointers, like std::unique_ptr and std::shared_ptr, which automatically manage memory and help prevent memory leaks.

Example (Smart Pointers in C++): 

#include <iostream>

#include <memory>


int main() {

    // Using std::unique_ptr for single ownership

    std::unique_ptr<int> numPtr = std::make_unique<int>(42);

    std::cout << "Dynamic Integer: " << *numPtr << std::endl;


    // Using std::shared_ptr for shared ownership

    std::shared_ptr<int> sharedNumPtr = std::make_shared<int>(42);

    std::cout << "Shared Dynamic Integer: " << *sharedNumPtr << std::endl;


    // Memory management is automatic - no need for explicit deallocation


    return 0;

}

Understanding dynamic memory allocation and memory management practices is crucial for developing efficient and reliable C and C++ programs. Practice using appropriate functions/operators and adopt smart pointers when applicable to improve code safety.