Ace Your DSA Interview: 30 Must-Know Question & Answer | AI-Powered Prep

Join our community to see how developers are using Workik AI everyday.

Top 25 DSA Interview Questions & Answers for Guaranteed Success

Q1: What is the difference between a stack and a queue in C++?

Stack:

Definition: A stack is a linear data structure that follows the Last In, First Out (LIFO) principle. Elements are added and removed from the same end, called the "top" of the stack.

Operations: Common operations include push() to add an element, pop() to remove the top element, and top() to access the top element without removing it.

Example in C++:

#include <stack>
std::stack<int> s;
s.push(10);  // Adds 10 to the stack
s.push(20);  // Adds 20 to the stack
int topElement = s.top();  // Accesses the top element (20)
s.pop();  // Removes the top element (20)

Queue:

Definition: A queue is a linear data structure that follows the First In, First Out (FIFO) principle. Elements are added at the rear and removed from the front.

Operations: Common operations include push() or enqueue() to add an element to the rear, pop() or dequeue() to remove an element from the front, and front() to access the front element.

Example in C++:

#include <queue>
std::queue<int> q;
q.push(10);  // Adds 10 to the queue
q.push(20);  // Adds 20 to the queue
int frontElement = q.front();  // Accesses the front element (10)
q.pop();  // Removes the front element (10)

Difference:

  • Order of Operations: A stack allows access to the most recently added element first (LIFO), while a queue allows access to the oldest added element first (FIFO).
  • Use Cases: Stacks are often used in backtracking algorithms, function call management, and parsing expressions, while queues are used in breadth-first search (BFS) algorithms, scheduling, and managing tasks in order.

Q2: Explain the concept of pointers in C++ and how they are used in data structures.

Pointer Definition: A pointer is a variable that stores the memory address of another variable. Pointers are a fundamental concept in C++ and are widely used in data structures to dynamically allocate memory, create complex structures like linked lists, trees, and graphs, and manage resources efficiently.

Pointer Declaration:

int x = 10;
int *ptr = &x;  // Pointer ptr stores the address of variable x

Usage in Data Structures:

  • Dynamic Memory Allocation: Pointers are used with operators like new and delete to allocate and deallocate memory dynamically.
int *arr = new int[10];  // Dynamically allocate an array of 10 integers
delete[] arr;  // Deallocate the array

Linked Lists: Pointers are crucial in implementing linked lists, where each node points to the next node.

struct Node {
    int data;
    Node* next;
    Node(int x) : data(x), next(nullptr) {}
};

Trees and Graphs: Pointers are used to create parent-child relationships in trees and connect vertices in graphs.

struct Node {
    int data;
    Node* left;
    Node* right;
    Node(int x) : data(x), left(nullptr), right(nullptr) {}
};

Importance in Data Structures:

  • Flexibility: Pointers allow for the creation of dynamic and flexible data structures that can grow and shrink in size as needed.
  • Efficient Memory Management: They enable efficient use of memory, especially when dealing with large or complex data structures.

Q3: What is the time complexity of searching for an element in a balanced binary search tree (BST) vs. an unbalanced BST?

Balanced BST:

Time Complexity: O(log n)

Explanation: In a balanced binary search tree (BST), the height of the tree is kept as small as possible, typically log(n) , where n is the number of nodes. This ensures that each comparison during a search operation significantly reduces the search space, leading to logarithmic time complexity.

Unbalanced BST:

Time Complexity: O(n) in the worst case

Explanation: If a BST becomes unbalanced, particularly if all elements are inserted in sorted order, the tree may degenerate into a linked list. In such a case, the height of the tree becomes n , and searching requires traversing each node, leading to linear time complexity.

Example in C++:

struct Node {
    int data;
    Node* left;
    Node* right;
    Node(int x) : data(x), left(nullptr), right(nullptr) {}
};

// Search function for BST
Node* search(Node* root, int key) {
    if (root == nullptr || root->data == key)
        return root;
    if (key < root->data)
        return search(root->left, key);
    return search(root->right, key);
}

Comparison:

  • Balanced BST: Provides efficient search operations, making it ideal for dynamic datasets where frequent updates occur.
  • Unbalanced BST: Can degrade to linear time complexity, making it inefficient for large datasets or when elements are inserted in a specific order.

Q4: Explain the difference between malloc() and new in C++.

malloc():

Definition: malloc() is a standard C library function used for dynamic memory allocation. It allocates a specified number of bytes and returns a pointer to the first byte of the allocated memory block.

Syntax:

int* ptr = (int*)malloc(sizeof(int) * 10);  // Allocates memory for an array of 10 integers

Initialization: malloc() does not initialize the allocated memory. The content of the allocated memory is indeterminate until explicitly initialized.

Deallocation: Memory allocated by malloc() must be deallocated using free() .

free(ptr);  // Deallocate memory

new:

Definition: new is a C++ operator that allocates memory and also calls the constructor to initialize the object.

Syntax:

int* ptr = new int[10];  // Allocates memory for an array of 10 integers and initializes them

Initialization: Memory allocated by new is automatically initialized by calling the constructor for objects or setting primitive types to their default values.

Deallocation: Memory allocated by new must be deallocated using delete or delete[] for arrays.

delete[] ptr;  // Deallocate memory

Differences:

  • Language Feature: malloc() is a C function, while new is a C++ operator.
  • Initialization: malloc() does not initialize memory, while new initializes memory by calling constructors for objects.
  • Type Safety: new returns a pointer of the correct type, whereas malloc() returns a void* , requiring explicit casting.
  • Error Handling: new throws an exception ( std::bad_alloc ) if allocation fails, while malloc() returns NULL .

Q5: Describe the concept of a "hash collision" in hash tables and how it can be resolved in C++.

Hash Collision:

Definition: A hash collision occurs when two different keys produce the same hash value, resulting in them being mapped to the same index in a hash table. Since the hash table uses the index to store key-value pairs, a collision must be resolved to store both entries correctly.

Resolution Techniques:

Chaining (Separate Chaining):

Concept: Each index in the hash table points to a linked list (or chain) of entries that hash to the same value. When a collision occurs, the new entry is added to the list at that index.

Example in C++:

#include <iostream>
#include <list>
#include <vector>

class HashTable {
private:
    std::vector<std::list<int>> table;
    int size;
public:
    HashTable(int s) : size(s) {
        table.resize(size);
    }
    int hashFunction(int key) {
        return key % size;
    }
    void insert(int key) {
        int index = hashFunction(key);
        table[index].push_back(key);
    }
    void display() {
        for (int i = 0; i < size; ++i) {
            std::cout << "Index " << i << ": ";
            for (auto& x : table[i]) {
                std::cout << x << " ";
            }
            std::cout << std::endl;
        }
    }
};

int main() {
    HashTable ht(10);
    ht.insert(15);
    ht.insert(25);
    ht.insert(35);
    ht.display();
    return 0;
}

Open Addressing:

Concept: Instead of using a linked list, open addressing resolves collisions by finding another open slot within the hash table array through probing (e.g., linear probing, quadratic probing, or double hashing).

Example in C++ (Linear Probing):

class OpenAddressingHashTable {
private:
    std::vector<int> table;
    int size;
    int emptySlot;
public:
    OpenAddressingHashTable(int s) : size(s), emptySlot(-1) {
        table.resize(size, emptySlot);
    }
    int hashFunction(int key) {
        return key % size;
    }
    void insert(int key) {
        int index = hashFunction(key);
        while (table[index] != emptySlot) {
            index = (index + 1) % size;
        }
        table[index] = key;
    }
    void display() {
        for (int i = 0; i < size; ++i) {
            if (table[i] != emptySlot) {
                std::cout << "Index " << i << ": " << table[i] << std::endl;
            }
        }
    }
};

int main() {
    OpenAddressingHashTable ht(10);
    ht.insert(15);
    ht.insert(25);
    ht.insert(35);
    ht.display();
    return 0;
}

Advantages and Disadvantages:

  • Chaining: Handles collisions efficiently even when the load factor is high, but requires additional memory for linked lists.

Request question

Please fill in the form below to submit your question.

Q6: Explain the concept of dynamic memory allocation in C++ and how it differs from static memory allocation.

Dynamic Memory Allocation:

Definition: Dynamic memory allocation in C++ refers to the process of allocating memory during the runtime of the program. It allows programs to request memory space as needed using the new operator and to release it using the delete operator.

Example:

int* ptr = new int;    // Allocates memory for a single integer
*ptr = 10;             // Assigns value to the allocated memory
delete ptr;            // Frees the allocated memory

Arrays:

int* arr = new int[10]; // Allocates memory for an array of 10 integers
delete[] arr;           // Frees the allocated array memory

Static Memory Allocation:

Definition: Static memory allocation occurs when memory is allocated at compile time. Variables declared using basic data types or arrays with a fixed size are statically allocated. The size of the memory allocated is fixed and cannot be changed during runtime.

Example:

int x = 10;  // Memory for x is allocated at compile time
int arr[10]; // Memory for an array of 10 integers is allocated at compile time

Differences:

  • Lifetime: Static memory is allocated at compile time and deallocated when the program exits, while dynamic memory is allocated at runtime and must be explicitly deallocated by the programmer.
  • Flexibility: Dynamic memory allows for the allocation of memory as needed, which is useful for creating data structures like linked lists, trees, and graphs that require flexible memory usage. Static memory allocation does not allow this flexibility.
  • Memory Management: In dynamic memory allocation, the programmer is responsible for managing memory using new and delete . In static memory allocation, the memory is automatically managed by the compiler.

Q7: What is recursion, and how does it differ from iteration in C++? Provide examples.

Recursion:

Definition: Recursion is a programming technique where a function calls itself directly or indirectly to solve a problem. Recursive functions typically have a base case to stop the recursion and one or more recursive cases to continue the function calls.

Example:

// Factorial function using recursion
int factorial(int n) {
    if (n <= 1) return 1;  // Base case
    return n * factorial(n - 1);  // Recursive case
}

Iteration:

Definition: Iteration is a technique where a set of instructions is repeated until a certain condition is met, typically using loops such as for , while , or do-while .

Example:

// Factorial function using iteration
int factorial(int n) {
    int result = 1;
    for (int i = 2; i <= n; ++i) {
        result *= i;
    }
    return result;
}

Differences:

  • Control Flow: In recursion, the control flow moves through repeated function calls, while in iteration, the control flow loops through a block of code multiple times.
  • Memory Usage: Recursion typically uses more memory than iteration due to the additional overhead of maintaining the call stack. Each recursive call adds a new frame to the stack, which can lead to stack overflow if the recursion depth is too great.
  • Efficiency: Iterative solutions are often more efficient in terms of time and space, as they avoid the overhead associated with recursive function calls. However, recursive solutions can be more elegant and easier to understand for problems that have a natural recursive structure, such as tree traversals or divide-and-conquer algorithms.

Q8: What is a binary search tree (BST), and how does it differ from a binary tree?

Binary Search Tree (BST):

Definition: A binary search tree (BST) is a special type of binary tree where each node has at most two children, and the left child's value is less than the parent's value, while the right child's value is greater than the parent's value.

Properties:

  • Left Subtree: Contains nodes with values less than the node's value.
  • Right Subtree: Contains nodes with values greater than the node's value.
  • No Duplicates: Typically, no two nodes have the same value.

Example in C++:

struct Node {
    int data;
    Node* left;
    Node* right;
    Node(int x) : data(x), left(nullptr), right(nullptr) {}
};

Binary Tree:

Definition: A binary tree is a general tree structure where each node has at most two children. There are no constraints on the values of the children relative to the parent node.

Example in C++:

struct Node {
    int data;
    Node* left;
    Node* right;
    Node(int x) : data(x), left(nullptr), right(nullptr) {}
};

Differences:

  • Order of Elements: In a BST, the elements are ordered such that for any node, all values in the left subtree are less than the node's value, and all values in the right subtree are greater. In a general binary tree, there is no such ordering, and the elements can be in any order.
  • Search Efficiency: The ordering of elements in a BST allows for efficient search operations with an average time complexity of O(log n) . In a general binary tree, search operations can take O(n) time since there is no inherent order.
  • Use Cases: BSTs are commonly used for searching, sorting, and dynamic data storage, while binary trees are used in scenarios where the structure or shape of the tree is more important than the order of the elements.

Q9: Explain the concept of tail recursion and how it differs from non-tail recursion in C++.

Tail Recursion:

Definition: Tail recursion is a special case of recursion where the recursive call is the last operation in the function. In tail-recursive functions, the result of the recursive call is immediately returned, and no further computation is needed after the call.

Example:

// Tail-recursive factorial function
int factorial(int n, int accumulator = 1) {
    if (n <= 1) return accumulator;  // Base case
    return factorial(n - 1, n * accumulator);  // Tail recursion
}

Non-Tail Recursion:

Definition: In non-tail recursion, the recursive call is not the last operation in the function. After the recursive call, the function may perform additional operations before returning the result.

Example:

// Non-tail-recursive factorial function
int factorial(int n) {
    if (n <= 1) return 1;  // Base case
    return n * factorial(n - 1);  // Non-tail recursion
}

Differences:

  • Optimization: Tail recursion can be optimized by the compiler into an iterative loop, reducing the overhead of function calls and stack space. This optimization is called "tail call optimization" (TCO). Non-tail recursion cannot be optimized in this way, as the additional operations after the recursive call require the current function's context to be maintained.
  • Efficiency: Tail-recursive functions are generally more efficient in terms of space because they can be converted into iterative loops, reducing the need for additional stack frames. Non-tail-recursive functions may lead to deep recursion and potential stack overflow for large inputs.

Q10: What is a graph, and what are the different ways to represent a graph in C++?

Graph:

Definition: A graph is a collection of nodes (also called vertices) and edges that connect pairs of nodes. Graphs can be directed (where edges have a direction) or undirected (where edges do not have a direction). They can also be weighted (where edges have a weight or cost) or unweighted.

Example of Graph Types:

  • Undirected Graph: Edges do not have a direction. If there is an edge between nodes A and B, you can traverse it from A to B or B to A.
  • Directed Graph (Digraph): Edges have a direction. If there is an edge from node A to node B, it can only be traversed in that direction.

Graph Representation in C++:

Adjacency Matrix:

Definition: An adjacency matrix is a 2D array of size VxV, where V is the number of vertices in the graph. The element at row i and column j is 1 (or the weight of the edge) if there is an edge from vertex i to vertex j ; otherwise, it is 0.

Example:

const int V = 4;
int graph[V][V] = {
    {0, 1, 0, 0},
    {1, 0, 1, 1},
    {0, 1, 0, 1},
    {0, 1, 1, 0}
};

Adjacency List:

Definition: An adjacency list is an array of lists. The array size is equal to the number of vertices, and each entry in the array contains a list of all the vertices adjacent to that vertex.

Example:

#include <iostream>
#include <vector>
using namespace std;

const int V = 4;
vector<int> adjList[V];

void addEdge(int u, int v) {
    adjList[u].push_back(v);
    adjList[v].push_back(u);  // For an undirected graph
}

int main() {
    addEdge(0, 1);
    addEdge(1, 2);
    addEdge(1, 3);
    addEdge(2, 3);

    // Output the adjacency list
    for (int i = 0; i < V; ++i) {
        cout << "Adjacency list of vertex " << i << ": ";
        for (int v : adjList[i]) {
            cout << v << " ";
        }
        cout << endl;
    }

    return 0;
}

Edge List:

Definition: An edge list is a list of all the edges in the graph. Each edge is represented as a pair (u, v) , where u and v are the vertices connected by the edge.

Example:

#include <iostream>
#include <vector>
using namespace std;

vector<pair<int, int>> edgeList;

void addEdge(int u, int v) {
    edgeList.push_back({u, v});
}

int main() {
    addEdge(0, 1);
    addEdge(1, 2);
    addEdge(1, 3);
    addEdge(2, 3);

    // Output the edge list
    for (auto edge : edgeList) {
        cout << "Edge: " << edge.first << " - " << edge.second << endl;
    }

    return 0;
}

Comparison:

  • Adjacency Matrix: Easy to implement and understand, but it consumes O(V^2) space, which can be inefficient for sparse graphs.
  • Adjacency List: More space-efficient, especially for sparse graphs, but slightly more complex to implement.
  • Edge List: Simple and space-efficient for storing edges, but not as efficient for checking the existence of an edge between two vertices.

Request question

Please fill in the form below to submit your question.

Q11: What is the significance of the "Big O" notation in analyzing algorithms?

Big O Notation:

Definition: Big O notation is a mathematical representation used to describe the upper bound of an algorithm's time complexity or space complexity. It provides an asymptotic analysis of the algorithm's performance as the input size ( n ) grows towards infinity.

Purpose: Big O notation helps in understanding the worst-case scenario of an algorithm's efficiency, allowing developers to compare the performance of different algorithms independently of hardware or software factors.

Common Big O Notations:

  • O(1): Constant time – The algorithm's runtime does not depend on the input size.
  • O(log n): Logarithmic time – The runtime grows logarithmically as the input size increases.
  • O(n): Linear time – The runtime grows linearly with the input size.
  • O(n log n): Linearithmic time – The runtime grows as n log n , typical for efficient sorting algorithms like mergesort and heapsort.
  • O(n^2): Quadratic time – The runtime grows quadratically, often seen in algorithms with nested loops, such as bubble sort.
  • O(2^n): Exponential time – The runtime doubles with each additional element in the input, often seen in recursive algorithms like the naive solution for the Traveling Salesman Problem.
  • O(n!): Factorial time – The runtime grows factorially, typical in algorithms that generate all permutations of a set.

Importance:

  • Predictability: Big O notation helps in predicting the scalability of algorithms as the input size increases.
  • Optimization: By analyzing the Big O notation, developers can optimize algorithms to improve their performance for large datasets.
  • Comparison: It provides a standardized way to compare the efficiency of different algorithms, making it easier to choose the best one for a particular problem.

Q12: What is a heap data structure, and what are the differences between a min-heap and a max-heap?

Heap:

Definition: A heap is a complete binary tree that satisfies the heap property. The heap can be either a min-heap or a max-heap based on the ordering of elements.

Complete Binary Tree: All levels of the tree are fully filled except possibly the last level, which is filled from left to right.

Min-Heap:

Definition: In a min-heap, the key at the root is the smallest among all keys present in the heap. The same property applies recursively to all nodes in the tree.

Properties:

  • The root node has the minimum value.
  • Every parent node is smaller than or equal to its children.

Use Case: Min-heaps are often used in priority queues where the smallest element is dequeued first.

#include <iostream>
#include <queue>
using namespace std;

int main() {
    priority_queue<int, vector<int>, greater<int>> minHeap;
    minHeap.push(3);
    minHeap.push(2);
    minHeap.push(15);
    cout << minHeap.top() << endl;  // Output: 2
    return 0;
}

Max-Heap:

Definition: In a max-heap, the key at the root is the largest among all keys present in the heap. The same property applies recursively to all nodes in the tree.

Properties:

  • The root node has the maximum value.
  • Every parent node is greater than or equal to its children.

Use Case: Max-heaps are used in algorithms like heapsort, and in applications where the maximum element needs to be accessed first.

#include <iostream>
#include <queue>
using namespace std;

int main() {
    priority_queue<int> maxHeap;
    maxHeap.push(3);
    maxHeap.push(2);
    maxHeap.push(15);
    cout << maxHeap.top() << endl;  // Output: 15
    return 0;
}

Differences:

  • Root Node: In a min-heap, the root node contains the smallest element, while in a max-heap, the root node contains the largest element.
  • Priority: Min-heaps prioritize the smallest elements, while max-heaps prioritize the largest elements.
  • Applications: Min-heaps are used in scenarios where the smallest element needs to be accessed frequently (e.g., Dijkstra's algorithm), while max-heaps are used in scenarios where the largest element is of interest (e.g., finding the k largest elements).

Q13: What is a hash table, and how is hashing used to store data efficiently in C++?

Hash Table:

Definition: A hash table is a data structure that maps keys to values using a hash function. It allows for efficient insertion, deletion, and lookup operations by computing an index (hash code) from the key and storing the value at that index in an array.

Hash Function:

Definition: A hash function takes an input (key) and returns an integer (hash code) that represents the index in the hash table where the corresponding value is stored.

Properties:

  • Deterministic: The same input always produces the same output.
  • Uniform Distribution: Ideally, the hash function distributes the keys uniformly across the hash table to minimize collisions.

Collisions:

Definition: A collision occurs when two different keys produce the same hash code, leading to a situation where both keys would be mapped to the same index in the hash table.

Resolution Techniques:

Chaining: Each index in the hash table points to a linked list of elements that hash to the same index. If a collision occurs, the new key-value pair is added to the list.

Open Addressing: When a collision occurs, the algorithm searches for the next available slot in the hash table according to a probing sequence (e.g., linear probing, quadratic probing).

Example of Hash Table in C++:

#include <iostream>
#include <unordered_map>
using namespace std;

int main() {
    unordered_map<string, int> hashTable;

    // Insert key-value pairs
    hashTable["apple"] = 5;
    hashTable["banana"] = 10;

    // Access elements
    cout << "apple: " << hashTable["apple"] << endl;
    cout << "banana: " << hashTable["banana"] << endl;

    return 0;
}

Efficiency:

  • Time Complexity: The average time complexity for insertion, deletion, and lookup operations in a hash table is O(1) . However, in the worst case, where collisions are not handled well, the time complexity can degrade to O(n) .
  • Space Complexity: The space complexity of a hash table is O(n) , where n is the number of key-value pairs stored in the table.

Applications:

  • Databases: Hash tables are used in database indexing to allow quick retrieval of data.
  • Caching: Hash tables are used to implement caches that store frequently accessed data for fast retrieval.
  • Symbol Tables: Compilers use hash tables to store and look up variable names and other symbols quickly.

Q14: What is an AVL tree, and how does it maintain balance?

AVL Tree:

Definition: An AVL tree is a self-balancing binary search tree (BST) where the difference between the heights of the left and right subtrees (balance factor) of any node is at most 1. It is named after its inventors, Adelson-Velsky and Landis.

Balance Factor: For every node in the tree, the balance factor is calculated as:

balanceFactor = height(leftSubtree) - height(rightSubtree);
The balance factor can be -1, 0, or 1.

Balancing Mechanism:

Rotations: When an insertion or deletion operation causes the tree to become unbalanced (i.e., the balance factor is not within the range [-1, 1]), the tree is rebalanced using one or more rotations.

Types of Rotations:

  • Left Rotation: Applied when the right subtree is higher (balance factor = -2) and the right subtree's right child is higher (right-right case).
  • Right Rotation: Applied when the left subtree is higher (balance factor = 2) and the left subtree's left child is higher (left-left case).
  • Left-Right Rotation: Applied when the left subtree is higher (balance factor = 2) and the left subtree's right child is higher (left-right case).
  • Right-Left Rotation: Applied when the right subtree is higher (balance factor = -2) and the right subtree's left child is higher (right-left case).

Example of Insertion in an AVL Tree:

struct Node {
    int data;
    Node* left;
    Node* right;
    int height;
    Node(int x) : data(x), left(nullptr), right(nullptr), height(1) {}
};

int height(Node* node) {
    return node ? node->height : 0;
}

int getBalanceFactor(Node* node) {
    return node ? height(node->left) - height(node->right) : 0;
}

Node* leftRotate(Node* x) {
    Node* y = x->right;
    Node* T2 = y->left;
    y->left = x;
    x->right = T2;
    x->height = max(height(x->left), height(x->right)) + 1;
    y->height = max(height(y->left), height(y->right)) + 1;
    return y;
}

Node* rightRotate(Node* y) {
    Node* x = y->left;
    Node* T2 = x->right;
    x->right = y;
    y->left = T2;
    y->height = max(height(y->left), height(y->right)) + 1;
    x->height = max(height(x->left), height(x->right)) + 1;
    return x;
}

Node* insert(Node* node, int key) {
    if (!node) return new Node(key);
    if (key < node->data) node->left = insert(node->left, key);
    else if (key > node->data) node->right = insert(node->right, key);
    else return node;

    node->height = 1 + max(height(node->left), height(node->right));
    int balance = getBalanceFactor(node);

    // Left Left Case
    if (balance > 1 && key < node->left->data) return rightRotate(node);

    // Right Right Case
    if (balance < -1 && key > node->right->data) return leftRotate(node);

    // Left Right Case
    if (balance > 1 && key > node->left->data) {
        node->left = leftRotate(node->left);
        return rightRotate(node);
    }

    // Right Left Case
    if (balance < -1 && key < node->right->data) {
        node->right = rightRotate(node->right);
        return leftRotate(node);
    }

    return node;
}

Advantages:

  • Efficiency: AVL trees provide guaranteed O(log n) time complexity for search, insertion, and deletion operations, even in the worst case, by maintaining balance.
  • Predictable Performance: The self-balancing property ensures that the tree's height remains logarithmic relative to the number of nodes, providing predictable performance.

Disadvantages:

  • Complexity: AVL trees are more complex to implement compared to simple binary search trees, due to the need for rotations during insertion and deletion operations.
  • Overhead: The balancing operations (rotations) introduce additional overhead during insertion and deletion, which may not be necessary for all applications.

Q15: What is the "divide and conquer" strategy, and how is it applied in algorithm design?

Divide and Conquer:

Definition: Divide and conquer is a strategy used in algorithm design where a problem is broken down into smaller subproblems of the same or related type, solved independently, and then combined to form a solution to the original problem.

Steps Involved:

  • Divide: Break the problem into smaller subproblems.
  • Conquer: Recursively solve each subproblem.
  • Combine: Combine the solutions to the subproblems to solve the original problem.

Examples of Divide and Conquer Algorithms:

Merge Sort:

  • Divide: Split the array into two halves.
  • Conquer: Recursively sort each half.
  • Combine: Merge the two sorted halves to produce the sorted array.

Time Complexity: O(n log n)

void merge(int arr[], int left, int mid, int right) {
    int n1 = mid - left + 1;
    int n2 = right - mid;
    int L[n1], R[n2];
    for (int i = 0; i < n1; i++) L[i] = arr[left + i];
    for (int j = 0; j < n2; j++) R[j] = arr[mid + 1 + j];
    int i = 0, j = 0, k = left;
    while (i < n1 && j < n2) {
        if (L[i] <= R[j]) arr[k++] = L[i++];
        else arr[k++] = R[j++];
    }
    while (i < n1) arr[k++] = L[i++];
    while (j < n2) arr[k++] = R[j++];
}

void mergeSort(int arr[], int left, int right) {
    if (left < right) {
        int mid = left + (right - left) / 2;
        mergeSort(arr, left, mid);
        mergeSort(arr, mid + 1, right);
        merge(arr, left, mid, right);
    }
}

Quick Sort:

  • Divide: Choose a pivot element and partition the array into two subarrays such that elements less than the pivot are on the left and elements greater than the pivot are on the right.
  • Conquer: Recursively sort the subarrays.
  • Combine: No explicit combination step is needed as the array is sorted in place.

Time Complexity: O(n log n) on average, but O(n^2) in the worst case.

int partition(int arr[], int low, int high) {
    int pivot = arr[high];
    int i = low - 1;
    for (int j = low; j < high; j++) {
        if (arr[j] <= pivot) {
            i++;
            swap(arr[i], arr[j]);
        }
    }
    swap(arr[i + 1], arr[high]);
    return i + 1;
}

void quickSort(int arr[], int low, int high) {
    if (low < high) {
        int pi = partition(arr, low, high);
        quickSort(arr, low, pi - 1);
        quickSort(arr, pi + 1, high);
    }
}

Advantages:

  • Efficiency: Divide and conquer algorithms often lead to more efficient solutions, particularly for large datasets, as they break down the problem into manageable parts.
  • Parallelism: The independent subproblems can often be solved in parallel, making divide and conquer algorithms well-suited for parallel computing environments.

Disadvantages:

  • Overhead: The process of dividing the problem and combining the solutions may introduce additional overhead, which may not be necessary for simpler problems.
  • Complexity: Designing divide and conquer algorithms can be more complex, especially in managing the recursive calls and combining the results.

Request question

Please fill in the form below to submit your question.

Q16: What is memoization in dynamic programming, and how does it improve the efficiency of recursive algorithms?

Memoization:

Definition: Memoization is a technique used in dynamic programming to improve the efficiency of recursive algorithms by storing the results of expensive function calls and reusing them when the same inputs occur again. It avoids the recomputation of subproblems by storing their results in a table (often a dictionary or array).

How It Works:

  • Recursive Function: You start with a recursive function that solves the problem by breaking it down into smaller subproblems.
  • Storage: A data structure, such as an array or a hash map, is used to store the results of subproblems.
  • Check: Before solving a subproblem, the algorithm checks if the result is already stored. If it is, the stored result is returned, avoiding the need for recalculation.

Example in C++:

#include <iostream>
#include <vector>
using namespace std;

vector<int> memo;

int fib(int n) {
    if (n <= 1) return n;
    if (memo[n] != -1) return memo[n];
    memo[n] = fib(n - 1) + fib(n - 2);
    return memo[n];
}

int main() {
    int n = 10;
    memo.resize(n + 1, -1);
    cout << "Fibonacci of " << n << " is " << fib(n) << endl;
    return 0;
}

Efficiency Improvement:

  • Without Memoization: A naive recursive algorithm may solve the same subproblem multiple times, leading to an exponential time complexity, such as O(2^n) in the case of calculating the Fibonacci sequence.
  • With Memoization: By storing the results of subproblems, memoization reduces the time complexity to O(n) , as each subproblem is solved only once.

Applications:

  • Dynamic Programming Problems: Memoization is commonly used in dynamic programming problems like the knapsack problem, longest common subsequence, and matrix chain multiplication.
  • Optimization Problems: It is used to optimize problems where the same subproblem is solved multiple times, reducing redundant calculations.

Q17: What is a topological sort, and in which scenarios is it used?

Topological Sort:

Definition: Topological sort is a linear ordering of the vertices of a directed acyclic graph (DAG) such that for every directed edge u → v , vertex u comes before vertex v in the ordering. A topological sort is only possible if the graph has no cycles (i.e., it must be a DAG).

Algorithm:

Depth-First Search (DFS) Based Approach:

  • Perform a DFS on the graph, keeping track of the visited vertices.
  • On the recursive callback (i.e., after visiting all descendants of a node), add the current vertex to the beginning of a list (or use a stack to maintain the order).
  • The list (or stack) will contain the vertices in topologically sorted order.
#include <iostream>
#include <vector>
#include <stack>
using namespace std;

void topologicalSortUtil(int v, vector<bool>& visited, stack<int>& Stack, vector<int> adj[]) {
    visited[v] = true;
    for (int i : adj[v]) {
        if (!visited[i]) {
            topologicalSortUtil(i, visited, Stack, adj);
        }
    }
    Stack.push(v);
}

void topologicalSort(int V, vector<int> adj[]) {
    stack<int> Stack;
    vector<bool> visited(V, false);
    for (int i = 0; i < V; i++) {
        if (!visited[i]) {
            topologicalSortUtil(i, visited, Stack, adj);
        }
    }
    while (!Stack.empty()) {
        cout << Stack.top() << " ";
        Stack.pop();
    }
}

int main() {
    int V = 6;
    vector<int> adj[V];
    adj[5].push_back(2);
    adj[5].push_back(0);
    adj[4].push_back(0);
    adj[4].push_back(1);
    adj[2].push_back(3);
    adj[3].push_back(1);

    cout << "Topological Sort of the given graph:\n";
    topologicalSort(V, adj);

    return 0;
}

Applications:

  • Task Scheduling: Topological sort is used to schedule tasks that have dependencies. For example, in a project with tasks that must be completed in a specific order, topological sorting helps determine the sequence of tasks.
  • Course Prerequisites: In academic settings, topological sorting is used to determine the order in which courses should be taken based on their prerequisites.
  • Build Systems: In software build systems, topological sort is used to determine the order in which modules or files should be compiled based on their dependencies.

Properties:

  • Uniqueness: A DAG can have more than one valid topological sort.
  • Cycle Detection: If a cycle is detected in the graph, a topological sort is not possible, indicating that the graph is not a DAG.

Q18: What is a trie, and how is it used for efficient string searching?

Trie:

Definition: A trie (also known as a prefix tree) is a tree-like data structure used to store a dynamic set of strings where each node represents a single character of a string. The path from the root to a particular node represents a prefix of a word.

Structure:

  • Nodes: Each node in the trie contains an array of pointers (or references) to its children, and a flag indicating if the node marks the end of a word.
  • Root: The root of the trie is an empty node that represents the start of all strings.
  • Edges: The edges of the trie are labeled with characters, and the nodes are not directly labeled with the strings they represent.

Operations:

  • Insertion: To insert a string into the trie, start at the root and for each character in the string, create a new child node (if it doesn't already exist) and move to that node. Mark the last node as the end of a word.
  • Search: To search for a string, start at the root and for each character in the string, move to the corresponding child node. If you reach the end of the string and the end of the word flag is set, the string is found.

Example in C++:

#include <iostream>
#include <unordered_map>
using namespace std;

struct TrieNode {
    unordered_map<char, TrieNode*> children;
    bool isEndOfWord;
    TrieNode() : isEndOfWord(false) {}
};

class Trie {
private:
    TrieNode*& root;
public:
    Trie() {
        root = new TrieNode();
    }

    void insert(string word) {
        TrieNode*& node = root;
        for (char c : word) {
            if (!node->children[c]) {
                node->children[c] = new TrieNode();
            }
            node = node->children[c];
        }
        node->isEndOfWord = true;
    }

    bool search(string word) {
        TrieNode*& node = root;
        for (char c : word) {
            if (!node->children[c]) return false;
            node = node->children[c];
        }
        return node->isEndOfWord;
    }
};

int main() {
    Trie trie;
    trie.insert("apple");
    trie.insert("app");
    cout << trie.search("apple") << endl;  // Output: 1 (true)
    cout << trie.search("app") << endl;    // Output: 1 (true)
    cout << trie.search("appl") << endl;   // Output: 0 (false)
    return 0;
}

Efficiency:

  • Time Complexity: The time complexity for inserting and searching a word in a trie is O(m) , where m is the length of the word. This is efficient compared to other data structures like hash tables or binary search trees, especially when dealing with a large set of strings with common prefixes.
  • Space Complexity: The space complexity can be higher compared to other data structures due to the storage of multiple pointers or references at each node. However, this is often mitigated by the shared prefixes.

Applications:

  • Autocomplete: Tries are used in search engines and text editors to implement autocomplete functionality by quickly finding words that start with a given prefix.
  • Spell Checking: Tries are used in spell checkers to store a dictionary of words and quickly check whether a word is valid.
  • Longest Prefix Matching: Tries are used in networking for IP routing (longest prefix match) to quickly find the route that matches the longest prefix of an IP address.

Q19: What is the difference between a depth-first search (DFS) and a breadth-first search (BFS) in graph traversal?

Depth-First Search (DFS):

Definition: DFS is a graph traversal algorithm that starts from a selected node (usually called the root) and explores as far as possible along each branch before backtracking. DFS uses a stack (either explicitly or via recursion) to keep track of the vertices to visit next.

Algorithm:

  • Start at the root node.
  • Mark the current node as visited and push it onto the stack.
  • Visit an adjacent, unvisited node, mark it as visited, and push it onto the stack.
  • Repeat until all vertices are visited, then backtrack.
  • Continue the process until all nodes have been explored.

Example in C++:

#include <iostream>
#include <vector>
#include <stack>
using namespace std;

void DFS(int v, vector<int> adj[], vector<bool>& visited) {
    stack<int> stack;
    stack.push(v);
    while (!stack.empty()) {
        int node = stack.top();
        stack.pop();
        if (!visited[node]) {
            cout << node << " ";
            visited[node] = true;
        }
        for (int i : adj[node]) {
            if (!visited[i]) {
                stack.push(i);
            }
        }
    }
}

int main() {
    int V = 4;
    vector<int> adj[V];
    adj[0].push_back(1);
    adj[0].push_back(2);
    adj[1].push_back(2);
    adj[2].push_back(0);
    adj[2].push_back(3);
    adj[3].push_back(3);

    vector<bool> visited(V, false);
    cout << "DFS starting from vertex 2:\n";
    DFS(2, adj, visited);

    return 0;
}

Breadth-First Search (BFS):

Definition: BFS is a graph traversal algorithm that starts from a selected node and explores all its neighbors before moving on to the next level of neighbors. BFS uses a queue to keep track of the vertices to visit next.

Algorithm:

  • Start at the root node.
  • Mark the current node as visited and enqueue it.
  • Dequeue a node from the front of the queue.
  • Visit all unvisited neighbors, mark them as visited, and enqueue them.
  • Repeat until the queue is empty.

Example in C++:

#include <iostream>
#include <vector>
#include <queue>
using namespace std;

void BFS(int v, vector<int> adj[], vector<bool>& visited) {
    queue<int> queue;
    visited[v] = true;
    queue.push(v);
    while (!queue.empty()) {
        int node = queue.front();
        queue.pop();
        cout << node << " ";
        for (int i : adj[node]) {
            if (!visited[i]) {
                visited[i] = true;
                queue.push(i);
            }
        }
    }
}

int main() {
    int V = 4;
    vector<int> adj[V];
    adj[0].push_back(1);
    adj[0].push_back(2);
    adj[1].push_back(2);
    adj[2].push_back(0);
    adj[2].push_back(3);
    adj[3].push_back(3);

    vector<bool> visited(V, false);
    cout << "BFS starting from vertex 2:\n";
    BFS(2, adj, visited);

    return 0;
}

Key Differences:

  • Order of Exploration:
    • DFS: Explores as deep as possible along each branch before backtracking, making it a good choice for searching in depth-first fashion.
    • BFS: Explores all neighbors level by level, making it ideal for finding the shortest path in unweighted graphs.
  • Data Structure:
    • DFS: Uses a stack (or recursion) to manage the vertices to be explored.
    • BFS: Uses a queue to manage the vertices to be explored.
  • Use Cases:
    • DFS: Suitable for problems like topological sorting, finding connected components, and solving puzzles like mazes.
    • BFS: Suitable for problems like finding the shortest path in an unweighted graph, level-order traversal of trees, and solving problems where all nodes at a given depth should be explored before moving to the next depth.

Q20: What is dynamic programming, and how does it differ from the divide and conquer strategy?

Dynamic Programming:

Definition: Dynamic programming (DP) is an optimization technique used to solve complex problems by breaking them down into simpler subproblems. It stores the results of subproblems to avoid redundant calculations and improve efficiency. DP is especially useful for problems with overlapping subproblems and optimal substructure properties.

Steps Involved:

  • Identify the Subproblems: Break the original problem into smaller, manageable subproblems.
  • Solve Subproblems: Solve each subproblem, storing the results in a table (array or hash map) to reuse them when needed.
  • Build the Solution: Combine the solutions of the subproblems to solve the original problem.

Difference from Divide and Conquer:

  • Overlapping Subproblems: Dynamic programming is designed to handle problems with overlapping subproblems, where the same subproblems are solved multiple times. In contrast, divide and conquer typically deals with non-overlapping subproblems, where each subproblem is solved independently.
  • Memoization vs. Recursion: DP often uses memoization or tabulation to store and reuse subproblem solutions, reducing redundant calculations. Divide and conquer primarily uses recursion to solve subproblems without storing intermediate results.
  • Optimal Substructure: Both dynamic programming and divide and conquer require the problem to have an optimal substructure, meaning the solution to the problem can be constructed from the solutions of its subproblems. However, DP explicitly stores these solutions to improve efficiency.

Example of Dynamic Programming:

Fibonacci Sequence: The Fibonacci sequence can be computed efficiently using dynamic programming by storing the results of previously computed values.

#include <iostream>
#include <vector>
using namespace std;

int fib(int n, vector<int>& dp) {
    if (n <= 1) return n;
    if (dp[n] != -1) return dp[n];
    dp[n] = fib(n - 1, dp) + fib(n - 2, dp);
    return dp[n];
}

int main() {
    int n = 10;
    vector<int> dp(n + 1, -1);
    cout << "Fibonacci of " << n << " is " << fib(n, dp) << endl;
    return 0;
}

Example of Divide and Conquer:

Merge Sort: Merge sort is a classic example of the divide and conquer strategy. The array is divided into two halves, each half is sorted recursively, and then the two sorted halves are merged.

void merge(int arr[], int left, int mid, int right) {
    int n1 = mid - left + 1;
    int n2 = right - mid;
    int L[n1], R[n2];
    for (int i = 0; i < n1; i++) L[i] = arr[left + i];
    for (int j = 0; j < n2; j++) R[j] = arr[mid + 1 + j];
    int i = 0, j = 0, k = left;
    while (i < n1 && j < n2) {
        if (L[i] <= R[j]) arr[k++] = L[i++];
        else arr[k++] = R[j++];
    }
    while (i < n1) arr[k++] = L[i++];
    while (j < n2) arr[k++] = R[j++];
}

void mergeSort(int arr[], int left, int right) {
    if (left < right) {
        int mid = left + (right - left) / 2;
        mergeSort(arr, left, mid);
        mergeSort(arr, mid + 1, right);
        merge(arr, left, mid, right);
    }
}

Request question

Please fill in the form below to submit your question.

Sharpen Your DSA Skills: 10 Hands-On Practical Questions & Answers

Q1: The following C++ code is intended to reverse an array, but it contains a bug. Identify and correct the error.
(Basic)
#include <iostream>
using namespace std;

void reverseArray(int arr[], int n) {
    int start = 0, end = n - 1;
    while (start < end) {
        arr[start] = arr[end];  // Error here
        arr[end] = arr[start];  // Error here
        start++;
        end--;
    }
}

int main() {
    int arr[] = {1, 2, 3, 4, 5};
    int n = sizeof(arr) / sizeof(arr[0]);
    reverseArray(arr, n);
    for (int i = 0; i < n; i++) {
        cout << arr[i] << " ";
    }
    return 0;
}

Error: The code mistakenly assigns arr[end] to arr[start] and then assigns the same value back to arr[end] . This results in both arr[start] and arr[end] holding the same value.

Corrected Code:

void reverseArray(int arr[], int n) {
    int start = 0, end = n - 1;
    while (start < end) {
        swap(arr[start], arr[end]);  // Corrected this line
        start++;
        end--;
    }
}
Q2: The following C++ code checks if a number is prime. Optimize it to improve performance.
(Basic)
#include <iostream>
using namespace std;

bool isPrime(int n) {
    if (n <= 1) return false;
    for (int i = 2; i < n; i++) {  // Inefficient loop
        if (n % i == 0) return false;
    }
    return true;
}

int main() {
    int num = 29;
    if (isPrime(num))
        cout << num << " is a prime number";
    else
        cout << num << " is not a prime number";
    return 0;
}

Inefficiency: The loop checks divisibility up to n-1 , which is unnecessary. The check can stop at the square root of n to reduce the number of iterations.

Optimized Code:

#include <cmath>

bool isPrime(int n) {
    if (n <= 1) return false;
    if (n == 2 || n == 3) return true;
    if (n % 2 == 0 || n % 3 == 0) return false;
    for (int i = 5; i * i <= n; i += 6) {
        if (n % i == 0 || n % (i + 2) == 0) return false;
    }
    return true;
}
Q3: What will be the output of the following C++ code?
(Basic)
#include <iostream>
using namespace std;

void func(int a, int b = 10, int c = 20) {
    cout << a << " " << b << " " << c << endl;
}

int main() {
    func(1);
    func(1, 2);
    func(1, 2, 3);
    return 0;
}

Output:

1 10 20
1 2 20
1 2 3

Explanation: The function func has default parameters b = 10 and c = 20 . When these parameters are not provided in the function call, the default values are used.

Q4: Write a function in C++ to find the first non-repeating character in a string. If all characters repeat, return '\0'.
(Intermediate)
#include <iostream>
#include <string>
using namespace std;

char firstNonRepeatingChar(const string& str) {
    // Your code here
}

int main() {
    string str = "swiss";
    cout << firstNonRepeatingChar(str) << endl;
    return 0;
}
#include <unordered_map>

char firstNonRepeatingChar(const string& str) {
    unordered_map<char, int> charCount;
    for (char c : str) {
        charCount[c]++;
    }
    for (char c : str) {
        if (charCount[c] == 1) {
            return c;
        }
    }
    return '\0';
}
Q5 (Intermediate): The following C++ code finds the maximum sum of a subarray in an array of integers. Optimize it for better performance.
(Intermediate)
#include <iostream>
using namespace std;

int maxSubArraySum(int arr[], int n) {
    int maxSum = INT_MIN;
    for (int i = 0; i < n; i++) {
        for (int j = i; j < n; j++) {
            int currentSum = 0;
            for (int k = i; k <= j; k++) {
                currentSum += arr[k];
            }
            if (currentSum > maxSum) {
                maxSum = currentSum;
            }
        }
    }
    return maxSum;
}

int main() {
    int arr[] = {-2, 1, -3, 4, -1, 2, 1, -5, 4};
    int n = sizeof(arr) / sizeof(arr[0]);
    cout << "Maximum sum of subarray: " << maxSubArraySum(arr, n);
    return 0;
}

Inefficiency: The code uses a triple nested loop, resulting in O(n^3) time complexity, which is inefficient.

Optimized Code: Use Kadane's Algorithm, which reduces the time complexity to O(n) .

int maxSubArraySum(int arr[], int n) {
    int maxSum = arr[0];
    int currentSum = arr[0];
    for (int i = 1; i < n; i++) {
        currentSum = max(arr[i], currentSum + arr[i]);
        maxSum = max(maxSum, currentSum);
    }
    return maxSum;
}
Q6: The following C++ code is intended to merge two sorted arrays into a single sorted array, but it contains a bug. Identify and correct the error.
(Intermediate)
#include <iostream>
using namespace std;

void mergeArrays(int arr1[], int arr2[], int n1, int n2, int arr3[]) {
    int i = 0, j = 0, k = 0;
    while (i < n1 && j < n2) {
        if (arr1[i] < arr2[j]) {
            arr3[k++] = arr1[i++];
        } else {
            arr3[k++] = arr2[j++];
        }
    }
    while (i < n1) {
        arr3[k++] = arr1[i++];  // Possible issue here
    }
    while (j < n2) {
        arr3[k++] = arr2[j++];  // Possible issue here
    }
}

int main() {
    int arr1[] = {1, 3, 5, 7};
    int arr2[] = {2, 4, 6, 8};
    int n1 = sizeof(arr1) / sizeof(arr1[0]);
    int n2 = sizeof(arr2) / sizeof(arr2[0]);
    int arr3[n1 + n2];
    mergeArrays(arr1, arr2, n1, n2, arr3);
    for (int i = 0; i < n1 + n2; i++) {
        cout << arr3[i] << " ";
    }
    return 0;
}

Error: The existing code is correct, but the prompt suggests that there might be an issue. The code merges two sorted arrays correctly.

No Correction Needed: The provided code works as intended, merging the two sorted arrays into a single sorted array.

Q7: The following C++ code calculates the factorial of a number using recursion. Optimize it to avoid stack overflow for large values of n.
(Advanced)
#include <iostream>
using namespace std;

int factorial(int n) {
    if (n <= 1) return 1;
    return n * factorial(n - 1);
}

int main() {
    int num = 20;
    cout << "Factorial of " << num << " is " << factorial(num);
    return 0;
}

Inefficiency: The recursive approach may lead to a stack overflow for large values of n due to deep recursion.

Optimized Code: Use an iterative approach to calculate the factorial, which avoids the risk of stack overflow.

int factorial(int n) {
    int result = 1;
    for (int i = 2; i <= n; i++) {
        result *= i;
    }
    return result;
}
Q8: Write a function in C++ to solve the "Two Sum" problem. Given an array of integers, return indices of the two numbers such that they add up to a specific target. Assume that each input would have exactly one solution, and you may not use the same element twice.
(Advanced)
#include <iostream>
#include <vector>
using namespace std;

pair<int, int> twoSum(const vector<int>& nums, int target) {
    // Your code here
}

int main() {
    vector<int> nums = {2, 7, 11, 15};
    int target = 9;
    pair<int, int> result = twoSum(nums, target);
    cout << "Indices: " << result.first << ", " << result.second << endl;
    return 0;
}
#include <unordered_map>

pair<int, int> twoSum(const vector<int>& nums, int target) {
    unordered_map<int, int> numMap;
    for (int i = 0; i < nums.size(); i++) {
        int complement = target - nums[i];
        if (numMap.find(complement) != numMap.end()) {
            return make_pair(numMap[complement], i);
        }
        numMap[nums[i]] = i;
    }
    return make_pair(-1, -1);  // Return -1, -1 if no solution is found
}
Q9: What will be the output of the following C++ code?
(Advanced)
#include <iostream>
using namespace std;

class Base {
public:
    virtual void show() {
        cout << "Base class" << endl;
    }
};

class Derived : public Base {
public:
    void show() {
        cout << "Derived class" << endl;
    }
};

int main() {
    Base* b = new Derived();
    b->show();
    delete b;
    return 0;
}

Output:

Derived class

Explanation: The show function is marked as virtual in the base class, enabling polymorphism. When the show function is called on a Base pointer pointing to a Derived object, the Derived class's show function is invoked.

Q10: The following C++ code checks whether a given string is a palindrome, considering only alphanumeric characters and ignoring cases. Optimize it for better performance.
(Advanced)
#include <iostream>
#include <string>
using namespace std;

bool isPalindrome(const string& s) {
    string filtered;
    for (char c : s) {
        if (isalnum(c)) {
            filtered += tolower(c);
        }
    }
    string reversed = filtered;
    reverse(reversed.begin(), reversed.end());
    return filtered == reversed;
}

int main() {
    string s = "A man, a plan, a canal: Panama";
    cout << (isPalindrome(s) ? "True" : "False");
    return 0;
}

Inefficiency: The code creates a new string filtered , then reverses it to compare with the original. This involves extra space and unnecessary processing.

Optimized Code: Use two pointers to compare characters from both ends of the string without creating a new string.

bool isPalindrome(const string& s) {
    int left = 0, right = s.size() - 1;
    while (left < right) {
        while (left < right && !isalnum(s[left])) left++;
        while (left < right && !isalnum(s[right])) right--;
        if (tolower(s[left]) != tolower(s[right])) return false;
        left++;
        right--;
    }
    return true;
}

Request question

Please fill in the form below to submit your question.

Unlock Your Potential in Full Stack Development – Use Workik AI Now!

Join developers who are using Workik’s AI assistance everyday for programming

Sign Up Now

Overview of DSA

What is DSA?

What are the popular use cases of DSA?

  • Search and Sorting Algorithms: Fundamental operations in computing.
  • Graph Algorithms: Used in social networks, navigation systems, etc.
  • Dynamic Programming: Solves complex problems by breaking them down.
  • Data Handling: Optimizing storage and retrieval in databases.

What are some of the tech roles requiring DSA expertise?

  • Software Engineer: Develops efficient software solutions.
  • Data Scientist: Analyzes and processes large datasets.
  • Algorithm Developer: Designs algorithms for specific applications.
  • Backend Developer: Implements efficient data handling and processing.

What pay package can be expected with expertise in DSA?

  • Junior Developer: $70,000 - $90,000 per year.
  • Mid-Level Developer: $90,000 - $120,000 per year.
  • Senior Developer: $110,000 - $150,000 per year.