🧱 Basic Data Structures
1. Arrays
      Definition: Contiguous memory allocation of elements of the same type.
      Time Complexity:
          o   Access: O(1)
          o   Insertion/Deletion (end): O(1)
          o   Insertion/Deletion (middle): O(n)
2. Stacks
      LIFO (Last In First Out)
      Operations:
          o   push(): Add element
          o   pop(): Remove top element
          o   peek(): View top element
      Applications:
          o   Expression parsing
          o   Backtracking (e.g., DFS, undo operations)
3. Queues
      FIFO (First In First Out)
      Operations:
          o   enqueue(): Add element
          o   dequeue(): Remove front element
      Variants:
          o   Deque (Double-ended Queue)
          o   Priority Queue (uses heap)
4. Linked Lists
      Singly Linked List: Each node has data + next
      Doubly Linked List: Each node has data + prev + next
      Circular Linked List: Last node points to first
      Advantages: Dynamic size, efficient insertion/deletion
      Disadvantages: No random access, O(n) traversal
🔍 Searching Algorithms
1. Linear Search
       Checks each element sequentially
       Time Complexity: O(n)
       Best Case: O(1), Worst Case: O(n)
2. Binary Search
       Used on sorted arrays
       Divide-and-conquer: Check mid, then left/right half
       Time Complexity: O(log n)
       Key Condition: Array must be sorted
python
CopyEdit
def binary_search(arr, key):
  l, r = 0, len(arr)-1
  while l <= r:
       mid = (l + r) // 2
       if arr[mid] == key:
         return mid
       elif arr[mid] < key:
         l = mid + 1
       else:
         r = mid - 1
  return -1
🌀 Sorting Algorithms
1. Bubble Sort
       Repeatedly swaps adjacent elements
       Time Complexity: O(n²)
       Space: O(1)
       Stable sort
2. Selection Sort
       Repeatedly finds minimum and places it at the front
       Time: O(n²)
      Space: O(1)
      Not stable
3. Insertion Sort
      Builds sorted array one element at a time
      Best case (sorted): O(n)
      Worst: O(n²)
      Stable
4. Merge Sort
      Divide & Conquer algorithm
      Recursively split array, merge sorted halves
      Time Complexity: O(n log n)
      Space Complexity: O(n)
      Stable
5. Quick Sort
      Select pivot, partition elements into < pivot and > pivot
      Best/Average: O(n log n)
      Worst (already sorted): O(n²)
      Not stable
🌐 Graph Algorithms
1. Graph Representations
      Adjacency Matrix: 2D array, O(V²) space
      Adjacency List: List of neighbors for each vertex, O(V + E) space
2. Breadth-First Search (BFS)
      Level-wise traversal using a queue
      Time Complexity: O(V + E)
      Applications:
           o   Shortest path (unweighted)
           o   Finding connected components
python
CopyEdit
from collections import deque
def bfs(graph, start):
  visited = set()
  queue = deque([start])
  while queue:
       node = queue.popleft()
       if node not in visited:
         print(node)
         visited.add(node)
         queue.extend(graph[node])
3. Depth-First Search (DFS)
       Explore as far as possible using recursion/stack
       Time Complexity: O(V + E)
       Applications:
            o   Topological sort
            o   Detecting cycles
            o   Path finding
python
ef dfs(graph, node, visited):
  if node not in visited:
       print(node)
       visited.add(node)
       for neighbor in graph[node]:
         dfs(graph, neighbor, visited)
4. Dijkstra’s Algorithm
       Find shortest path in weighted graph (no negative edges)
       Uses a priority queue (min-heap)
       Time Complexity: O((V + E) log V)
5. Topological Sort
       Only for DAGs (Directed Acyclic Graphs)
       Orders tasks respecting dependencies
       Performed using DFS or Kahn’s Algorithm
📊 Comparison Table
             Time
Algorithm                     Best Use
             Complexity
Binary                        Search sorted
             O(log n)
Search                        data
                              Stable, large
Merge Sort O(n log n)
                              data
Quick Sort   O(n log n)       Fast avg case
BFS / DFS    O(V + E)         Graph traversal
                              Weighted
Dijkstra’s   O((V+E) log V)
                              graphs
Topo Sort    O(V + E)         Task scheduling