PART 1: Structured Data & Objects

What is Structured Data?

Structured data organizes information into well-defined formats so programs can store, access, and manipulate it efficiently.

Key categories:

Type Description Ruby Example
Primitive Single values 42, "hello", true
Composite Groups of values Arrays, Hashes, Structs
Abstract Logical structure Stack, Queue, Tree

Structured data is the foundation of every data structure we study.


Objects – Attributes and Methods

An object bundles data (attributes) and behavior (methods) together.

class BankAccount
  # Attributes
  attr_accessor :owner, :balance

  def initialize(owner, balance = 0)
    @owner   = owner     # instance variable = attribute
    @balance = balance
  end

  # Methods (behavior)
  def deposit(amount)
    @balance += amount
  end

  def withdraw(amount)
    @balance -= amount if amount <= @balance
  end

  def to_s
    "#{@owner}: $#{@balance}"
  end
end

acct = BankAccount.new("Alice", 500)
acct.deposit(200)
puts acct   # => Alice: $700

Records and Fields

A record is a composite data structure with named fields — similar to a row in a database table.

In Ruby, use Struct for lightweight records:

# Define a record type with named fields
Student = Struct.new(:name, :id, :gpa)

# Create instances (records)
s1 = Student.new("Alice",  1001, 3.9)
s2 = Student.new("Bob",    1002, 3.4)
s3 = Student.new("Carol",  1003, 3.7)

puts s1.name  # => Alice
puts s1.gpa   # => 3.9

# Structs support iteration over fields
s1.each_pair { |field, val| puts "#{field}: #{val}" }
# name: Alice
# id: 1001
# gpa: 3.9

Key idea: Records map directly to objects with fixed attributes.


Memory Allocation

How does Ruby reserve space for data?

Stack Memory          Heap Memory
+------------+         +---------------------+
|  x = 5     | ------> |  Integer object: 5  |
|  s = "hi"  | ------> |  String object: "hi"|
|  arr = []  |-------> |  Array object: []   |
+------------+         +---------------------+
  Variables              Actual Objects
  (references)           (allocated data)
x   = 42          # stack holds reference; Fixnum on heap
str = "hello"     # stack ref → String object on heap
arr = [1, 2, 3]   # stack ref → Array object on heap

puts x.object_id    # unique heap address identifier
puts str.object_id
puts arr.object_id

In Ruby, almost everything lives on the heap. The stack holds references (pointers) to objects.


PART 2: Static vs Dynamic Structures

Static Memory Structures

Static structures have a fixed size determined at compile time or program start.

# Fixed-size array (static-style in concept)
DAYS_OF_WEEK = Array.new(7)   # allocated once, size = 7
DAYS_OF_WEEK[0] = "Monday"
DAYS_OF_WEEK[6] = "Sunday"

# Constants and fixed tables are classic static structures
PRIMES = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29].freeze

# .freeze prevents modification — enforces static behavior
# PRIMES << 31  # => RuntimeError: can't modify frozen Array

Characteristics of static structures:

  • Size is fixed
  • Memory is predictable and fast to access
  • No overhead from memory management
  • Waste space if not fully used

Dynamic Structures – Heap Allocation

Dynamic structures grow and shrink at runtime on the heap.

# Ruby Array is fully dynamic
list = []               # starts empty

10.times { |i| list << i * 2 }   # grows automatically
puts list.inspect
# => [0, 2, 4, 6, 8, 10, 12, 14, 16, 18]

list.delete_if { |x| x > 10 }    # shrinks automatically
puts list.inspect
# => [0, 2, 4, 6, 8, 10]

# Hash also grows dynamically
inventory = {}
inventory["apples"]  = 42
inventory["bananas"] = 17
inventory["mangoes"] = 5
puts inventory

Heap allocation means Ruby requests memory from the OS as needed and releases it when objects are no longer reachable.


Garbage Collection and Recycling

Ruby uses automatic garbage collection (GC) — you never call free().

# When an object loses all references, it becomes garbage
def create_garbage
  temp = "I am temporary"   # allocated on heap
  arr  = [1, 2, 3, 4, 5]   # allocated on heap
  # temp and arr go out of scope here → eligible for GC
end

create_garbage   # after this call, those objects are garbage

# Ruby uses a tri-color mark-and-sweep + generational GC
GC.start         # manually trigger (rarely needed)
puts GC.stat[:heap_live_slots]   # objects currently alive

How GC works: 1. Mark — trace all reachable objects from root references 2. Sweep — reclaim memory of unmarked (unreachable) objects 3. Compact (Ruby 3+) — defragment heap for better locality


PART 3: Linked Lists

Singly Linked Lists

Each node stores data + a pointer to the next node only.

class Node
  attr_accessor :data, :next_node
  def initialize(data)
    @data      = data
    @next_node = nil
  end
end

class SinglyLinkedList
  def initialize
    @head = nil
  end

  def prepend(data)
    node      = Node.new(data)
    node.next_node = @head
    @head     = node
  end

  def to_s
    result, cur = [], @head
    while cur
      result << cur.data
      cur = cur.next_node
    end
    result.join(" -> ") + " -> nil"
  end
end

list = SinglyLinkedList.new
[3, 2, 1].each { |v| list.prepend(v) }
puts list   # => 1 -> 2 -> 3 -> nil

Doubly Linked Lists

Each node has pointers to both next and previous nodes.

class DNode
  attr_accessor :data, :prev, :next_node
  def initialize(data)
    @data = data; @prev = nil; @next_node = nil
  end
end

class DoublyLinkedList
  def initialize
    @head = @tail = nil
  end

  def append(data)
    node = DNode.new(data)
    if @tail
      @tail.next_node = node
      node.prev       = @tail
    else
      @head = node
    end
    @tail = node
  end

  def traverse_backward
    cur = @tail
    while cur
      print "#{cur.data} "
      cur = cur.prev
    end
    puts
  end
end

dll = DoublyLinkedList.new
[10, 20, 30].each { |v| dll.append(v) }
dll.traverse_backward   # => 30 20 10

Linked List Operations

Insertion, Deletion, Traversal — the three core operations:

# INSERTION at position
def insert_after(target_data, new_data)
  cur = @head
  while cur
    if cur.data == target_data
      node           = Node.new(new_data)
      node.next_node = cur.next_node
      cur.next_node  = node
      return
    end
    cur = cur.next_node
  end
end

# DELETION by value  O(n)
def delete(data)
  return if @head.nil?
  if @head.data == data
    @head = @head.next_node; return
  end
  cur = @head
  while cur.next_node
    if cur.next_node.data == data
      cur.next_node = cur.next_node.next_node; return
    end
    cur = cur.next_node
  end
end

# TRAVERSAL  O(n)
def traverse
  cur = @head
  while cur
    print "#{cur.data} "
    cur = cur.next_node
  end
end

Arrays vs Hashes

# ARRAY — ordered, integer-indexed  O(1) access by index
fruits = ["apple", "banana", "cherry"]
puts fruits[0]          # => apple  (O(1))
puts fruits.length      # => 3
fruits.push("date")     # append   O(1) amortized

# HASH — key-value pairs, O(1) average lookup
ages = { "Alice" => 30, "Bob" => 25, "Carol" => 28 }
puts ages["Alice"]      # => 30   (O(1) avg)
ages["Dave"] = 22
ages.each { |name, age| puts "#{name} is #{age}" }
Feature Array Hash
Index type Integer (0-based) Any object
Access time O(1) by index O(1) avg by key
Order Maintained Maintained (Ruby 1.9+)
Use case Sequences Lookup tables / maps
Memory Compact Higher overhead

PART 4: Sorting Algorithms

Bubble Sort

Repeatedly swaps adjacent out-of-order elements. O(n²) — simple but slow.

def bubble_sort(arr)
  n = arr.length
  loop do
    swapped = false
    (n - 1).times do |i|
      if arr[i] > arr[i + 1]
        arr[i], arr[i + 1] = arr[i + 1], arr[i]   # swap
        swapped = true
      end
    end
    break unless swapped   # early exit if already sorted
  end
  arr
end

data = [64, 34, 25, 12, 22, 11, 90]
puts bubble_sort(data).inspect
# => [11, 12, 22, 25, 34, 64, 90]

Complexity: Time O(n²) best/worst | Space O(1)


Insertion Sort

Builds the sorted array one element at a time. O(n²) worst, O(n) best (nearly sorted).

def insertion_sort(arr)
  (1...arr.length).each do |i|
    key = arr[i]
    j   = i - 1
    while j >= 0 && arr[j] > key
      arr[j + 1] = arr[j]   # shift right
      j -= 1
    end
    arr[j + 1] = key          # insert
  end
  arr
end

data = [12, 11, 13, 5, 6]
puts insertion_sort(data).inspect
# => [5, 6, 11, 12, 13]

Best for: Small datasets or nearly sorted data.

Complexity: Time O(n²) worst | O(n) best | Space O(1)


Merge Sort

Divide-and-conquer. Split in half, sort each half, merge. O(n log n) guaranteed.

def merge_sort(arr)
  return arr if arr.length <= 1

  mid   = arr.length / 2
  left  = merge_sort(arr[0...mid])
  right = merge_sort(arr[mid..])

  merge(left, right)
end

def merge(left, right)
  result = []
  until left.empty? || right.empty?
    if left.first <= right.first
      result << left.shift
    else
      result << right.shift
    end
  end
  result + left + right
end

data = [38, 27, 43, 3, 9, 82, 10]
puts merge_sort(data).inspect
# => [3, 9, 10, 27, 38, 43, 82]

Quick Sort

Pick a pivot, partition around it, recurse. O(n log n) average, O(n²) worst.

def quicksort(arr)
  return arr if arr.length <= 1

  pivot = arr[arr.length / 2]         # choose middle as pivot
  left  = arr.select { |x| x < pivot }
  mid   = arr.select { |x| x == pivot }
  right = arr.select { |x| x > pivot }

  quicksort(left) + mid + quicksort(right)
end

data = [3, 6, 8, 10, 1, 2, 1]
puts quicksort(data).inspect
# => [1, 1, 2, 3, 6, 8, 10]
Algorithm Best Average Worst Space
Bubble O(n) O(n²) O(n²) O(1)
Insertion O(n) O(n²) O(n²) O(1)
Merge O(n log n) O(n log n) O(n log n) O(n)
Quick O(n log n) O(n log n) O(n²) O(log n)

PART 5: Stacks

Stacks – LIFO Structure

A stack is Last-In, First-Out. Think: stack of plates.

class Stack
  def initialize
    @data = []
  end

  def push(item)  = @data.push(item)
  def pop         = @data.pop
  def peek        = @data.last
  def empty?      = @data.empty?
  def size        = @data.size
  def to_s        = @data.inspect
end

s = Stack.new
s.push(10); s.push(20); s.push(30)
puts s          # => [10, 20, 30]
puts s.peek     # => 30  (top, not removed)
puts s.pop      # => 30  (removed)
puts s          # => [10, 20]

Use cases: Order reversal, undo operations, function call frames, expression evaluation.


Stacks – Order Reversal & Functions

Stacks reverse order and manage function call frames:

# Order reversal using a stack
def reverse_string(str)
  stack = Stack.new
  str.each_char { |c| stack.push(c) }
  result = ""
  result += stack.pop until stack.empty?
  result
end
puts reverse_string("hello")  # => "olleh"

# How the call stack works during function calls:
# Each call pushes a "frame" with local vars + return address
def factorial(n)          # frame pushed
  return 1 if n <= 1
  n * factorial(n - 1)    # new frame pushed recursively
end                       # frame popped on return

puts factorial(5)   # => 120
# Call stack at deepest point:
# [factorial(1), factorial(2), factorial(3), factorial(4), factorial(5)]

Recursion

Recursion uses the call stack implicitly — each call adds a new stack frame.

# Every recursive solution needs:
# 1. BASE CASE  — stops recursion
# 2. RECURSIVE CASE — moves toward base case

# Fibonacci with recursion
def fib(n)
  return n if n <= 1          # base case
  fib(n - 1) + fib(n - 2)    # recursive case
end

puts (0..7).map { |i| fib(i) }.inspect
# => [0, 1, 1, 2, 3, 5, 8, 13]

# Tower of Hanoi — classic recursion example
def hanoi(n, from, to, via)
  if n == 1
    puts "Move disk 1: #{from} → #{to}"
    return
  end
  hanoi(n - 1, from, via, to)
  puts "Move disk #{n}: #{from} → #{to}"
  hanoi(n - 1, via, to, from)
end

hanoi(3, "A", "C", "B")

PART 6: Queues

Queues – FIFO Structure

A queue is First-In, First-Out. Think: checkout line.

class Queue
  def initialize
    @data = []
  end

  def enqueue(item) = @data.push(item)      # add to back
  def dequeue       = @data.shift           # remove from front
  def front         = @data.first
  def empty?        = @data.empty?
  def size          = @data.size
  def to_s          = @data.inspect
end

q = Queue.new
q.enqueue("Alice"); q.enqueue("Bob"); q.enqueue("Carol")
puts q            # => ["Alice", "Bob", "Carol"]
puts q.dequeue    # => "Alice"
puts q            # => ["Bob", "Carol"]

Use cases: Task scheduling, BFS traversal, print spoolers, buffering.


Circular Queue

A circular queue reuses empty slots by wrapping the tail pointer around.

class CircularQueue
  def initialize(capacity)
    @cap   = capacity
    @data  = Array.new(capacity)
    @head  = 0
    @tail  = 0
    @size  = 0
  end

  def enqueue(item)
    raise "Queue full" if @size == @cap
    @data[@tail] = item
    @tail = (@tail + 1) % @cap   # wrap around!
    @size += 1
  end

  def dequeue
    raise "Queue empty" if @size == 0
    item  = @data[@head]
    @head = (@head + 1) % @cap   # wrap around!
    @size -= 1
    item
  end
end

cq = CircularQueue.new(3)
cq.enqueue(1); cq.enqueue(2); cq.enqueue(3)
puts cq.dequeue   # => 1
cq.enqueue(4)     # reuses slot 0

Priority Queue

Elements are dequeued in priority order, not arrival order.

class PriorityQueue
  def initialize
    @data = []   # store [priority, item] pairs
  end

  def enqueue(item, priority)
    @data << [priority, item]
    @data.sort_by! { |p, _| -p }  # highest priority first
  end

  def dequeue
    @data.shift&.last
  end

  def peek
    @data.first&.last
  end
end

pq = PriorityQueue.new
pq.enqueue("low task",      1)
pq.enqueue("critical task", 10)
pq.enqueue("normal task",   5)

puts pq.dequeue   # => "critical task"   (priority 10)
puts pq.dequeue   # => "normal task"     (priority 5)
puts pq.dequeue   # => "low task"        (priority 1)

PART 7: Trees

Binary Trees

A binary tree has nodes with at most two children (left and right).

class TreeNode
  attr_accessor :val, :left, :right
  def initialize(val)
    @val = val; @left = nil; @right = nil
  end
end

# Build a Binary Search Tree (BST)
class BST
  def insert(root, val)
    return TreeNode.new(val) if root.nil?
    if val < root.val
      root.left  = insert(root.left,  val)
    else
      root.right = insert(root.right, val)
    end
    root
  end

  def inorder(root)   # Left → Root → Right (sorted output)
    return if root.nil?
    inorder(root.left)
    print "#{root.val} "
    inorder(root.right)
  end
end

bst  = BST.new
root = nil
[5, 3, 7, 1, 4, 6, 8].each { |v| root = bst.insert(root, v) }
bst.inorder(root)   # => 1 3 4 5 6 7 8

Math Expression Tree

Expression trees represent mathematical expressions structurally.

class ExprNode
  attr_accessor :val, :left, :right
  def initialize(val, left = nil, right = nil)
    @val = val; @left = left; @right = right
  end
end

# Tree for: (3 + 4) * (5 - 2)
#        *
#       / \
#      +   -
#     / \ / \
#    3  4 5  2

tree = ExprNode.new("*",
         ExprNode.new("+", ExprNode.new(3), ExprNode.new(4)),
         ExprNode.new("-", ExprNode.new(5), ExprNode.new(2)))

def evaluate(node)
  return node.val if node.left.nil?   # leaf = number
  l = evaluate(node.left)
  r = evaluate(node.right)
  case node.val
  when "+" then l + r
  when "-" then l - r
  when "*" then l * r
  when "/" then l / r.to_f
  end
end

puts evaluate(tree)   # => 21  [ (3+4)*(5-2) = 7*3 ]

B+ Trees

B+ Trees are balanced multi-way trees used in databases and file systems.

Structure of a B+ Tree (order 3):

Internal nodes: [ 10 | 20 ]
               /     |      \
         [5|7]    [12|17]   [22|30]
          ↓          ↓          ↓
       Leaf nodes (linked) → faster range queries

Key properties:
  • All data lives in LEAF nodes
  • Internal nodes are index-only
  • Leaf nodes are linked (→) for range scans
  • Guaranteed O(log n) search, insert, delete
# Conceptual B+ Tree search in Ruby
class BPlusNode
  attr_accessor :keys, :children, :is_leaf, :next_leaf
  def initialize(is_leaf = false)
    @keys = []; @children = []; @is_leaf = is_leaf
    @next_leaf = nil
  end
end

# B+ Trees are used in: MySQL InnoDB, PostgreSQL indexes,
# file systems (NTFS, ext4), and many NoSQL stores.
puts "B+ Tree ensures O(log n) operations regardless of size"

PART 8: Graphs

Graphs – Mazes

A graph is a set of nodes (vertices) connected by edges. Mazes are graphs!

# Represent a maze as an adjacency list
maze = {
  "start" => ["A", "B"],
  "A"     => ["C"],
  "B"     => ["C", "D"],
  "C"     => ["end"],
  "D"     => ["end"],
  "end"   => []
}

# BFS to find shortest path through maze
def bfs(graph, start, goal)
  queue   = [[start, [start]]]
  visited = {}
  until queue.empty?
    node, path = queue.shift
    return path if node == goal
    next if visited[node]
    visited[node] = true
    graph[node].each { |n| queue << [n, path + [n]] }
  end
  nil
end

path = bfs(maze, "start", "end")
puts path.join(" → ")   # => start → A → C → end

Transition Matrix (Adjacency Matrix)

A transition matrix represents graph edges as a 2D grid.

# Graph:  A→B, A→C, B→C, C→D
#         0=A, 1=B, 2=C, 3=D
matrix = [
  [0, 1, 1, 0],   # A connects to B, C
  [0, 0, 1, 0],   # B connects to C
  [0, 0, 0, 1],   # C connects to D
  [0, 0, 0, 0]    # D: no outgoing edges
]

LABELS = %w[A B C D]

# Print adjacency
matrix.each_with_index do |row, i|
  row.each_with_index do |val, j|
    puts "#{LABELS[i]} → #{LABELS[j]}" if val == 1
  end
end

# Check if path A→C exists: matrix[0][2]
puts "A→C edge exists? #{matrix[0][2] == 1}"   # => true
# Check if path B→A exists: matrix[1][0]
puts "B→A edge exists? #{matrix[1][0] == 1}"   # => false

Dijkstra’s Algorithm

Finds the shortest path in a weighted graph with non-negative weights.

def dijkstra(graph, start)
  dist    = Hash.new(Float::INFINITY)
  dist[start] = 0
  visited = {}
  pq      = [[0, start]]   # [cost, node]

  until pq.empty?
    cost, node = pq.min_by { |c, _| c }
    pq.delete([cost, node])
    next if visited[node]
    visited[node] = true

    (graph[node] || []).each do |neighbor, weight|
      new_cost = cost + weight
      if new_cost < dist[neighbor]
        dist[neighbor] = new_cost
        pq << [new_cost, neighbor]
      end
    end
  end
  dist
end

graph = {
  "A" => [["B", 1], ["C", 4]],
  "B" => [["C", 2], ["D", 5]],
  "C" => [["D", 1]],
  "D" => []
}
puts dijkstra(graph, "A").inspect
# => {"A"=>0, "B"=>1, "C"=>3, "D"=>4}

Flow Analysis (Max-Flow)

Flow networks model capacity-constrained systems (networks, pipelines, traffic).

# Ford-Fulkerson Max-Flow (BFS-based Edmonds-Karp)
def bfs_path(capacity, source, sink, parent)
  visited = { source => true }
  queue   = [source]
  until queue.empty?
    u = queue.shift
    capacity[u]&.each do |v, cap|
      if !visited[v] && cap > 0
        visited[v] = true
        parent[v]  = u
        return true if v == sink
        queue << v
      end
    end
  end
  false
end

# Key concepts for flow analysis:
# • Source node (supply) → Sink node (demand)
# • Each edge has a capacity constraint
# • Max-Flow = Min-Cut (Ford-Fulkerson theorem)
# • Applications: network routing, matching, scheduling

puts "Max-Flow applications:"
puts "  • Internet traffic routing"
puts "  • Supply chain optimization"
puts "  • Bipartite matching (job assignments)"
puts "  • Image segmentation"

EXAM SUMMARY

Quick Reference – Complexity Cheat Sheet

Structure / Algorithm Access Search Insert Delete Space
Array (static) O(1) O(n) O(n) O(n) O(n)
Hash / Dictionary O(1) O(1) O(1) O(1) O(n)
Singly Linked List O(n) O(n) O(1)† O(1)† O(n)
Doubly Linked List O(n) O(n) O(1)† O(1)† O(n)
Stack (push/pop) O(n) O(n) O(1) O(1) O(n)
Queue (enq/deq) O(n) O(n) O(1) O(1) O(n)
BST (balanced) O(log n) O(log n) O(log n) O(log n) O(n)
B+ Tree O(log n) O(log n) O(log n) O(log n) O(n)
Bubble Sort O(1) worst O(n²)
Merge Sort O(n) always O(n log n)
Quick Sort O(log n) avg O(n log n)

† With direct reference to node


Key Concepts to Remember

Memory: - Stack = references/primitives | Heap = objects - GC marks reachable objects, sweeps the rest

Lists: Singly (forward only) vs Doubly (forward + backward)

Trees: - BST: left < root < right → inorder gives sorted output - B+ Tree: all data in leaves, leaves linked for range queries - Expression Tree: post-order traversal = evaluate

Graphs: - Adjacency List: sparse graphs (less memory) - Adjacency Matrix: dense graphs (O(1) edge lookup) - Dijkstra: greedy shortest path (non-negative weights only)

Stacks: LIFO → reversal, recursion, function calls

Queues: FIFO → BFS, scheduling | Circular → reuse space | Priority → ordered service

Good luck on your final exam!

---
title: "Data Structures & Algorithms"
author: "R Batzinger"
date: "`r Sys.Date()`"
output:
  output: html_notebook
link-citations: TRUE
subtitle: Comprehensive Final Exam Review
---

```{r setup, include=FALSE}
knitr::opts_chunk$set(echo = TRUE)
```

---

# PART 1: Structured Data & Objects {.section-title}

## What is Structured Data?

**Structured data** organizes information into well-defined formats so programs can store, access, and manipulate it efficiently.

Key categories:

| Type | Description | Ruby Example |
|------|-------------|--------------|
| **Primitive** | Single values | `42`, `"hello"`, `true` |
| **Composite** | Groups of values | Arrays, Hashes, Structs |
| **Abstract** | Logical structure | Stack, Queue, Tree |

> Structured data is the foundation of every data structure we study.

---

## Objects – Attributes and Methods

An **object** bundles **data** (attributes) and **behavior** (methods) together.

```ruby
class BankAccount
  # Attributes
  attr_accessor :owner, :balance

  def initialize(owner, balance = 0)
    @owner   = owner     # instance variable = attribute
    @balance = balance
  end

  # Methods (behavior)
  def deposit(amount)
    @balance += amount
  end

  def withdraw(amount)
    @balance -= amount if amount <= @balance
  end

  def to_s
    "#{@owner}: $#{@balance}"
  end
end

acct = BankAccount.new("Alice", 500)
acct.deposit(200)
puts acct   # => Alice: $700
```

---

## Records and Fields

A **record** is a composite data structure with named **fields** — similar to a row in a database table.

In Ruby, use `Struct` for lightweight records:

```ruby
# Define a record type with named fields
Student = Struct.new(:name, :id, :gpa)

# Create instances (records)
s1 = Student.new("Alice",  1001, 3.9)
s2 = Student.new("Bob",    1002, 3.4)
s3 = Student.new("Carol",  1003, 3.7)

puts s1.name  # => Alice
puts s1.gpa   # => 3.9

# Structs support iteration over fields
s1.each_pair { |field, val| puts "#{field}: #{val}" }
# name: Alice
# id: 1001
# gpa: 3.9
```

> **Key idea:** Records map directly to objects with fixed attributes.

---

## Memory Allocation

**How does Ruby reserve space for data?**

```
Stack Memory          Heap Memory
+------------+         +---------------------+
|  x = 5     | ------> |  Integer object: 5  |
|  s = "hi"  | ------> |  String object: "hi"|
|  arr = []  |-------> |  Array object: []   |
+------------+         +---------------------+
  Variables              Actual Objects
  (references)           (allocated data)
```

```ruby
x   = 42          # stack holds reference; Fixnum on heap
str = "hello"     # stack ref → String object on heap
arr = [1, 2, 3]   # stack ref → Array object on heap

puts x.object_id    # unique heap address identifier
puts str.object_id
puts arr.object_id
```

> In Ruby, **almost everything lives on the heap**. The stack holds *references* (pointers) to objects.

---

# PART 2: Static vs Dynamic Structures {.section-title}

## Static Memory Structures

**Static** structures have a **fixed size** determined at compile time or program start.

```ruby
# Fixed-size array (static-style in concept)
DAYS_OF_WEEK = Array.new(7)   # allocated once, size = 7
DAYS_OF_WEEK[0] = "Monday"
DAYS_OF_WEEK[6] = "Sunday"

# Constants and fixed tables are classic static structures
PRIMES = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29].freeze

# .freeze prevents modification — enforces static behavior
# PRIMES << 31  # => RuntimeError: can't modify frozen Array
```

**Characteristics of static structures:**

- Size is fixed
- Memory is predictable and fast to access
- No overhead from memory management
- Waste space if not fully used

---

## Dynamic Structures – Heap Allocation

**Dynamic** structures grow and shrink at **runtime** on the **heap**.

```ruby
# Ruby Array is fully dynamic
list = []               # starts empty

10.times { |i| list << i * 2 }   # grows automatically
puts list.inspect
# => [0, 2, 4, 6, 8, 10, 12, 14, 16, 18]

list.delete_if { |x| x > 10 }    # shrinks automatically
puts list.inspect
# => [0, 2, 4, 6, 8, 10]

# Hash also grows dynamically
inventory = {}
inventory["apples"]  = 42
inventory["bananas"] = 17
inventory["mangoes"] = 5
puts inventory
```

> **Heap allocation** means Ruby requests memory from the OS as needed and releases it when objects are no longer reachable.

---

## Garbage Collection and Recycling

Ruby uses **automatic garbage collection (GC)** — you never call `free()`.

```ruby
# When an object loses all references, it becomes garbage
def create_garbage
  temp = "I am temporary"   # allocated on heap
  arr  = [1, 2, 3, 4, 5]   # allocated on heap
  # temp and arr go out of scope here → eligible for GC
end

create_garbage   # after this call, those objects are garbage

# Ruby uses a tri-color mark-and-sweep + generational GC
GC.start         # manually trigger (rarely needed)
puts GC.stat[:heap_live_slots]   # objects currently alive
```

**How GC works:**
1. **Mark** — trace all reachable objects from root references
2. **Sweep** — reclaim memory of unmarked (unreachable) objects
3. **Compact** *(Ruby 3+)* — defragment heap for better locality

---

# PART 3: Linked Lists {.section-title}

## Singly Linked Lists

Each **node** stores data + a pointer to the **next** node only.

```ruby
class Node
  attr_accessor :data, :next_node
  def initialize(data)
    @data      = data
    @next_node = nil
  end
end

class SinglyLinkedList
  def initialize
    @head = nil
  end

  def prepend(data)
    node      = Node.new(data)
    node.next_node = @head
    @head     = node
  end

  def to_s
    result, cur = [], @head
    while cur
      result << cur.data
      cur = cur.next_node
    end
    result.join(" -> ") + " -> nil"
  end
end

list = SinglyLinkedList.new
[3, 2, 1].each { |v| list.prepend(v) }
puts list   # => 1 -> 2 -> 3 -> nil
```

---

## Doubly Linked Lists

Each node has pointers to **both** next and previous nodes.

```ruby
class DNode
  attr_accessor :data, :prev, :next_node
  def initialize(data)
    @data = data; @prev = nil; @next_node = nil
  end
end

class DoublyLinkedList
  def initialize
    @head = @tail = nil
  end

  def append(data)
    node = DNode.new(data)
    if @tail
      @tail.next_node = node
      node.prev       = @tail
    else
      @head = node
    end
    @tail = node
  end

  def traverse_backward
    cur = @tail
    while cur
      print "#{cur.data} "
      cur = cur.prev
    end
    puts
  end
end

dll = DoublyLinkedList.new
[10, 20, 30].each { |v| dll.append(v) }
dll.traverse_backward   # => 30 20 10
```

---

## Linked List Operations

**Insertion, Deletion, Traversal** — the three core operations:

```ruby
# INSERTION at position
def insert_after(target_data, new_data)
  cur = @head
  while cur
    if cur.data == target_data
      node           = Node.new(new_data)
      node.next_node = cur.next_node
      cur.next_node  = node
      return
    end
    cur = cur.next_node
  end
end

# DELETION by value  O(n)
def delete(data)
  return if @head.nil?
  if @head.data == data
    @head = @head.next_node; return
  end
  cur = @head
  while cur.next_node
    if cur.next_node.data == data
      cur.next_node = cur.next_node.next_node; return
    end
    cur = cur.next_node
  end
end

# TRAVERSAL  O(n)
def traverse
  cur = @head
  while cur
    print "#{cur.data} "
    cur = cur.next_node
  end
end
```

---

## Arrays vs Hashes

```ruby
# ARRAY — ordered, integer-indexed  O(1) access by index
fruits = ["apple", "banana", "cherry"]
puts fruits[0]          # => apple  (O(1))
puts fruits.length      # => 3
fruits.push("date")     # append   O(1) amortized

# HASH — key-value pairs, O(1) average lookup
ages = { "Alice" => 30, "Bob" => 25, "Carol" => 28 }
puts ages["Alice"]      # => 30   (O(1) avg)
ages["Dave"] = 22
ages.each { |name, age| puts "#{name} is #{age}" }
```

| Feature | Array | Hash |
|---------|-------|------|
| Index type | Integer (0-based) | Any object |
| Access time | O(1) by index | O(1) avg by key |
| Order | Maintained | Maintained (Ruby 1.9+) |
| Use case | Sequences | Lookup tables / maps |
| Memory | Compact | Higher overhead |

---

# PART 4: Sorting Algorithms {.section-title}

## Bubble Sort

Repeatedly swaps adjacent out-of-order elements. **O(n²)** — simple but slow.

```ruby
def bubble_sort(arr)
  n = arr.length
  loop do
    swapped = false
    (n - 1).times do |i|
      if arr[i] > arr[i + 1]
        arr[i], arr[i + 1] = arr[i + 1], arr[i]   # swap
        swapped = true
      end
    end
    break unless swapped   # early exit if already sorted
  end
  arr
end

data = [64, 34, 25, 12, 22, 11, 90]
puts bubble_sort(data).inspect
# => [11, 12, 22, 25, 34, 64, 90]
```

**Complexity:** Time O(n²) best/worst | Space O(1)

---

## Insertion Sort

Builds the sorted array one element at a time. **O(n²)** worst, **O(n)** best (nearly sorted).

```ruby
def insertion_sort(arr)
  (1...arr.length).each do |i|
    key = arr[i]
    j   = i - 1
    while j >= 0 && arr[j] > key
      arr[j + 1] = arr[j]   # shift right
      j -= 1
    end
    arr[j + 1] = key          # insert
  end
  arr
end

data = [12, 11, 13, 5, 6]
puts insertion_sort(data).inspect
# => [5, 6, 11, 12, 13]
```

**Best for:** Small datasets or nearly sorted data.

**Complexity:** Time O(n²) worst | O(n) best | Space O(1)

---

## Merge Sort

Divide-and-conquer. Split in half, sort each half, merge. **O(n log n)** guaranteed.

```ruby
def merge_sort(arr)
  return arr if arr.length <= 1

  mid   = arr.length / 2
  left  = merge_sort(arr[0...mid])
  right = merge_sort(arr[mid..])

  merge(left, right)
end

def merge(left, right)
  result = []
  until left.empty? || right.empty?
    if left.first <= right.first
      result << left.shift
    else
      result << right.shift
    end
  end
  result + left + right
end

data = [38, 27, 43, 3, 9, 82, 10]
puts merge_sort(data).inspect
# => [3, 9, 10, 27, 38, 43, 82]
```

---

## Quick Sort

Pick a pivot, partition around it, recurse. **O(n log n)** average, **O(n²)** worst.

```ruby
def quicksort(arr)
  return arr if arr.length <= 1

  pivot = arr[arr.length / 2]         # choose middle as pivot
  left  = arr.select { |x| x < pivot }
  mid   = arr.select { |x| x == pivot }
  right = arr.select { |x| x > pivot }

  quicksort(left) + mid + quicksort(right)
end

data = [3, 6, 8, 10, 1, 2, 1]
puts quicksort(data).inspect
# => [1, 1, 2, 3, 6, 8, 10]
```

| Algorithm | Best | Average | Worst | Space |
|-----------|------|---------|-------|-------|
| Bubble    | O(n) | O(n²) | O(n²) | O(1) |
| Insertion | O(n) | O(n²) | O(n²) | O(1) |
| Merge     | O(n log n) | O(n log n) | O(n log n) | O(n) |
| Quick     | O(n log n) | O(n log n) | O(n²) | O(log n) |

---

# PART 5: Stacks {.section-title}

## Stacks – LIFO Structure

A **stack** is Last-In, First-Out. Think: stack of plates.

```ruby
class Stack
  def initialize
    @data = []
  end

  def push(item)  = @data.push(item)
  def pop         = @data.pop
  def peek        = @data.last
  def empty?      = @data.empty?
  def size        = @data.size
  def to_s        = @data.inspect
end

s = Stack.new
s.push(10); s.push(20); s.push(30)
puts s          # => [10, 20, 30]
puts s.peek     # => 30  (top, not removed)
puts s.pop      # => 30  (removed)
puts s          # => [10, 20]
```

**Use cases:** Order reversal, undo operations, function call frames, expression evaluation.

---

## Stacks – Order Reversal & Functions

Stacks **reverse order** and manage **function call frames**:

```ruby
# Order reversal using a stack
def reverse_string(str)
  stack = Stack.new
  str.each_char { |c| stack.push(c) }
  result = ""
  result += stack.pop until stack.empty?
  result
end
puts reverse_string("hello")  # => "olleh"

# How the call stack works during function calls:
# Each call pushes a "frame" with local vars + return address
def factorial(n)          # frame pushed
  return 1 if n <= 1
  n * factorial(n - 1)    # new frame pushed recursively
end                       # frame popped on return

puts factorial(5)   # => 120
# Call stack at deepest point:
# [factorial(1), factorial(2), factorial(3), factorial(4), factorial(5)]
```

---

## Recursion

**Recursion** uses the call stack implicitly — each call adds a new stack frame.

```ruby
# Every recursive solution needs:
# 1. BASE CASE  — stops recursion
# 2. RECURSIVE CASE — moves toward base case

# Fibonacci with recursion
def fib(n)
  return n if n <= 1          # base case
  fib(n - 1) + fib(n - 2)    # recursive case
end

puts (0..7).map { |i| fib(i) }.inspect
# => [0, 1, 1, 2, 3, 5, 8, 13]

# Tower of Hanoi — classic recursion example
def hanoi(n, from, to, via)
  if n == 1
    puts "Move disk 1: #{from} → #{to}"
    return
  end
  hanoi(n - 1, from, via, to)
  puts "Move disk #{n}: #{from} → #{to}"
  hanoi(n - 1, via, to, from)
end

hanoi(3, "A", "C", "B")
```

---

# PART 6: Queues {.section-title}

## Queues – FIFO Structure

A **queue** is First-In, First-Out. Think: checkout line.

```ruby
class Queue
  def initialize
    @data = []
  end

  def enqueue(item) = @data.push(item)      # add to back
  def dequeue       = @data.shift           # remove from front
  def front         = @data.first
  def empty?        = @data.empty?
  def size          = @data.size
  def to_s          = @data.inspect
end

q = Queue.new
q.enqueue("Alice"); q.enqueue("Bob"); q.enqueue("Carol")
puts q            # => ["Alice", "Bob", "Carol"]
puts q.dequeue    # => "Alice"
puts q            # => ["Bob", "Carol"]
```

**Use cases:** Task scheduling, BFS traversal, print spoolers, buffering.

---

## Circular Queue

A **circular queue** reuses empty slots by wrapping the tail pointer around.

```ruby
class CircularQueue
  def initialize(capacity)
    @cap   = capacity
    @data  = Array.new(capacity)
    @head  = 0
    @tail  = 0
    @size  = 0
  end

  def enqueue(item)
    raise "Queue full" if @size == @cap
    @data[@tail] = item
    @tail = (@tail + 1) % @cap   # wrap around!
    @size += 1
  end

  def dequeue
    raise "Queue empty" if @size == 0
    item  = @data[@head]
    @head = (@head + 1) % @cap   # wrap around!
    @size -= 1
    item
  end
end

cq = CircularQueue.new(3)
cq.enqueue(1); cq.enqueue(2); cq.enqueue(3)
puts cq.dequeue   # => 1
cq.enqueue(4)     # reuses slot 0
```

---

## Priority Queue

Elements are dequeued in **priority order**, not arrival order.

```ruby
class PriorityQueue
  def initialize
    @data = []   # store [priority, item] pairs
  end

  def enqueue(item, priority)
    @data << [priority, item]
    @data.sort_by! { |p, _| -p }  # highest priority first
  end

  def dequeue
    @data.shift&.last
  end

  def peek
    @data.first&.last
  end
end

pq = PriorityQueue.new
pq.enqueue("low task",      1)
pq.enqueue("critical task", 10)
pq.enqueue("normal task",   5)

puts pq.dequeue   # => "critical task"   (priority 10)
puts pq.dequeue   # => "normal task"     (priority 5)
puts pq.dequeue   # => "low task"        (priority 1)
```

---

# PART 7: Trees {.section-title}

## Binary Trees

A **binary tree** has nodes with at most **two children** (left and right).

```ruby
class TreeNode
  attr_accessor :val, :left, :right
  def initialize(val)
    @val = val; @left = nil; @right = nil
  end
end

# Build a Binary Search Tree (BST)
class BST
  def insert(root, val)
    return TreeNode.new(val) if root.nil?
    if val < root.val
      root.left  = insert(root.left,  val)
    else
      root.right = insert(root.right, val)
    end
    root
  end

  def inorder(root)   # Left → Root → Right (sorted output)
    return if root.nil?
    inorder(root.left)
    print "#{root.val} "
    inorder(root.right)
  end
end

bst  = BST.new
root = nil
[5, 3, 7, 1, 4, 6, 8].each { |v| root = bst.insert(root, v) }
bst.inorder(root)   # => 1 3 4 5 6 7 8
```

---

## Math Expression Tree

Expression trees represent **mathematical expressions** structurally.

```ruby
class ExprNode
  attr_accessor :val, :left, :right
  def initialize(val, left = nil, right = nil)
    @val = val; @left = left; @right = right
  end
end

# Tree for: (3 + 4) * (5 - 2)
#        *
#       / \
#      +   -
#     / \ / \
#    3  4 5  2

tree = ExprNode.new("*",
         ExprNode.new("+", ExprNode.new(3), ExprNode.new(4)),
         ExprNode.new("-", ExprNode.new(5), ExprNode.new(2)))

def evaluate(node)
  return node.val if node.left.nil?   # leaf = number
  l = evaluate(node.left)
  r = evaluate(node.right)
  case node.val
  when "+" then l + r
  when "-" then l - r
  when "*" then l * r
  when "/" then l / r.to_f
  end
end

puts evaluate(tree)   # => 21  [ (3+4)*(5-2) = 7*3 ]
```

---

## B+ Trees

**B+ Trees** are balanced multi-way trees used in databases and file systems.

```
Structure of a B+ Tree (order 3):

Internal nodes: [ 10 | 20 ]
               /     |      \
         [5|7]    [12|17]   [22|30]
          ↓          ↓          ↓
       Leaf nodes (linked) → faster range queries

Key properties:
  • All data lives in LEAF nodes
  • Internal nodes are index-only
  • Leaf nodes are linked (→) for range scans
  • Guaranteed O(log n) search, insert, delete
```

```ruby
# Conceptual B+ Tree search in Ruby
class BPlusNode
  attr_accessor :keys, :children, :is_leaf, :next_leaf
  def initialize(is_leaf = false)
    @keys = []; @children = []; @is_leaf = is_leaf
    @next_leaf = nil
  end
end

# B+ Trees are used in: MySQL InnoDB, PostgreSQL indexes,
# file systems (NTFS, ext4), and many NoSQL stores.
puts "B+ Tree ensures O(log n) operations regardless of size"
```

---

# PART 8: Graphs {.section-title}

## Graphs – Mazes

A **graph** is a set of **nodes (vertices)** connected by **edges**. Mazes are graphs!

```ruby
# Represent a maze as an adjacency list
maze = {
  "start" => ["A", "B"],
  "A"     => ["C"],
  "B"     => ["C", "D"],
  "C"     => ["end"],
  "D"     => ["end"],
  "end"   => []
}

# BFS to find shortest path through maze
def bfs(graph, start, goal)
  queue   = [[start, [start]]]
  visited = {}
  until queue.empty?
    node, path = queue.shift
    return path if node == goal
    next if visited[node]
    visited[node] = true
    graph[node].each { |n| queue << [n, path + [n]] }
  end
  nil
end

path = bfs(maze, "start", "end")
puts path.join(" → ")   # => start → A → C → end
```

---

## Transition Matrix (Adjacency Matrix)

A **transition matrix** represents graph edges as a 2D grid.

```ruby
# Graph:  A→B, A→C, B→C, C→D
#         0=A, 1=B, 2=C, 3=D
matrix = [
  [0, 1, 1, 0],   # A connects to B, C
  [0, 0, 1, 0],   # B connects to C
  [0, 0, 0, 1],   # C connects to D
  [0, 0, 0, 0]    # D: no outgoing edges
]

LABELS = %w[A B C D]

# Print adjacency
matrix.each_with_index do |row, i|
  row.each_with_index do |val, j|
    puts "#{LABELS[i]} → #{LABELS[j]}" if val == 1
  end
end

# Check if path A→C exists: matrix[0][2]
puts "A→C edge exists? #{matrix[0][2] == 1}"   # => true
# Check if path B→A exists: matrix[1][0]
puts "B→A edge exists? #{matrix[1][0] == 1}"   # => false
```

---

## Dijkstra's Algorithm

Finds the **shortest path** in a weighted graph with non-negative weights.

```ruby
def dijkstra(graph, start)
  dist    = Hash.new(Float::INFINITY)
  dist[start] = 0
  visited = {}
  pq      = [[0, start]]   # [cost, node]

  until pq.empty?
    cost, node = pq.min_by { |c, _| c }
    pq.delete([cost, node])
    next if visited[node]
    visited[node] = true

    (graph[node] || []).each do |neighbor, weight|
      new_cost = cost + weight
      if new_cost < dist[neighbor]
        dist[neighbor] = new_cost
        pq << [new_cost, neighbor]
      end
    end
  end
  dist
end

graph = {
  "A" => [["B", 1], ["C", 4]],
  "B" => [["C", 2], ["D", 5]],
  "C" => [["D", 1]],
  "D" => []
}
puts dijkstra(graph, "A").inspect
# => {"A"=>0, "B"=>1, "C"=>3, "D"=>4}
```

---

## Flow Analysis (Max-Flow)

**Flow networks** model capacity-constrained systems (networks, pipelines, traffic).

```ruby
# Ford-Fulkerson Max-Flow (BFS-based Edmonds-Karp)
def bfs_path(capacity, source, sink, parent)
  visited = { source => true }
  queue   = [source]
  until queue.empty?
    u = queue.shift
    capacity[u]&.each do |v, cap|
      if !visited[v] && cap > 0
        visited[v] = true
        parent[v]  = u
        return true if v == sink
        queue << v
      end
    end
  end
  false
end

# Key concepts for flow analysis:
# • Source node (supply) → Sink node (demand)
# • Each edge has a capacity constraint
# • Max-Flow = Min-Cut (Ford-Fulkerson theorem)
# • Applications: network routing, matching, scheduling

puts "Max-Flow applications:"
puts "  • Internet traffic routing"
puts "  • Supply chain optimization"
puts "  • Bipartite matching (job assignments)"
puts "  • Image segmentation"
```

---

# EXAM SUMMARY {.section-title}

## Quick Reference – Complexity Cheat Sheet

| Structure / Algorithm | Access | Search | Insert | Delete | Space |
|-----------------------|--------|--------|--------|--------|-------|
| Array (static)        | O(1)   | O(n)   | O(n)   | O(n)   | O(n)  |
| Hash / Dictionary     | O(1)   | O(1)   | O(1)   | O(1)   | O(n)  |
| Singly Linked List    | O(n)   | O(n)   | O(1)†  | O(1)†  | O(n)  |
| Doubly Linked List    | O(n)   | O(n)   | O(1)†  | O(1)†  | O(n)  |
| Stack (push/pop)      | O(n)   | O(n)   | O(1)   | O(1)   | O(n)  |
| Queue (enq/deq)       | O(n)   | O(n)   | O(1)   | O(1)   | O(n)  |
| BST (balanced)        | O(log n) | O(log n) | O(log n) | O(log n) | O(n) |
| B+ Tree               | O(log n) | O(log n) | O(log n) | O(log n) | O(n) |
| Bubble Sort           | —      | —      | —      | —      | O(1) worst O(n²) |
| Merge Sort            | —      | —      | —      | —      | O(n) always O(n log n) |
| Quick Sort            | —      | —      | —      | —      | O(log n) avg O(n log n) |

† With direct reference to node

---

## Key Concepts to Remember

**Memory:**
- Stack = references/primitives | Heap = objects
- GC marks reachable objects, sweeps the rest

**Lists:** Singly (forward only) vs Doubly (forward + backward)

**Trees:**
- BST: left < root < right → inorder gives sorted output
- B+ Tree: all data in leaves, leaves linked for range queries
- Expression Tree: post-order traversal = evaluate

**Graphs:**
- Adjacency List: sparse graphs (less memory)
- Adjacency Matrix: dense graphs (O(1) edge lookup)
- Dijkstra: greedy shortest path (non-negative weights only)

**Stacks:** LIFO → reversal, recursion, function calls

**Queues:** FIFO → BFS, scheduling | Circular → reuse space | Priority → ordered service

> **Good luck on your final exam!** 
