[WIP] The Algorithm Design Manual

Pet Peeve: The author sometimes skips steps without an explaination (like the integer truncation in "stop and think: incremental correctness"). Some examples are hard to follow for an international student (like understanding the lottery system in "war story: pschic modeling").

Chapter 1: Introduction to Algorithm Design

A computational problem is a specification of a desired input-output relationship. e.g.

Computational problem: Sorting

Input: A sequence of nn values a1a_1, ..., ana_n.

Output: The permutation (reordering) of the input sequence such that a1a2a'_1 \leq a'_2 ... an1ana'_{n-1} \leq a'_n.

An instance of a problem is all the inputs needed to compute a solution to the problem. Alternatively, a computational problem is the set of all (problem) instances and the desired output. e.g.

To sort the permutation {8, 3, 6, 7} is an instance of the general sorting problem and {3, 6, 7, 8} is the desired output.

An algorithm is a well-defined computational procedure that halts with a desired output when given any instance of the given computational problem. An algorithm is an abstract (idea) and can have multiple implementations (programs). e.g.

/*
Insertion sort is an algorithm to the sorting problem: Start with a single
element (thus forming a trivially sorted list) and then incrementally inserts
the remaining elements so that the list stays sorted.
*/
insertion_sort(item s[], int n)
{
  int i,j; /* counters */
  for (i=0; i<n; i++) {
    j=i;
    
    while ((j>0) && (s[j] < s[j-1])) {
      swap(&s[j],&s[j-1]);
      j = j-1;
    }
  }
}

Modeling the problem

Real-world applications involve real-world objects. However, most algorithms are designed to work on rigorously defined abstract structures. Modeling is the art of formulating your application in terms of procedures on such fundamental abstract structures so as to exploit the existing algorithms literature.

Determining that you are dealing with an instance of a general problem allows you to solve it using general well-known algorithms.

Certain problems can be modeled in several different ways, some much better than others. Modeling is only the first step in designing an algorithm for a problem. Be alert but not dismissive for how the details of your applications differ from a candidate model.

Combinatorial Objects

  1. Permutations are arrangements, or orderings, of items. E.g: {1,4,3,2} and {4,3,2,1} are two distinct permutations of the same set of four integers.
  2. Subsets represent selections from a set of items. E.g: {1,3,4} and {2} are two distinct subsets of the first four integers. Order does not matter in subsets the way it does with permutations.
  3. Trees represent hierarchical relationships between items.
  4. Graphs represent relationships between arbitrary pairs of objects.
  5. Points define locations in some geometric space.
  6. Polygons define regions in some geometric spaces.
  7. Strings represent sequences of characters, or patterns.

Modeling real-world structures with trees and graphs

Recursive Objects

Learning to think recursively is learning to look for big things that are made from smaller things of exactly the same type as the big thing.
If you think of houses as sets of rooms, then adding or deleting a room still leaves a house behind.

  1. Permutations: Delete the first element of a permutation of n things {1, ..., n} and you get a permutation of the remaining n-1 things. Basis case: {}
  2. Subsets: Every subset of {1, ..., n} contains a subset of (1, ..., n - 1) obtained by deleting element n. Basis case: {}
  3. Trees: Delete the root of a tree and you get a collection of smaller trees. Delete any leaf of a tree and you get a slightly smaller tree. Basis case: 1 vertex.
  4. Graphs: Delete any vertex from a graph, and you get a smaller graph. Now divide the vertices of a graph into two groups, left and right. Cut through all edges that span from left to right, and you get two smaller graphs, and a bunch of broken edges. Basis case: 1 vertex.
  5. Point sets: Take a cloud of points, and separate them into two groups by drawing a line. Now you have two smaller clouds of points. Basis case: 1 point.
  6. Polygons: Inserting any internal chord between two non-adjacent vertices of a simple polygon cuts it into two smaller polygons. Basis case: triangle.
  7. Strings: Delete the first character from a string, and you get a shorter string. Basis case: empty string.

Recursive descriptions of objects require both decomposition rules and basis cases, namely the specification of the smallest and simplest objects where the decomposition stops.

Recursive decompositions of combinatorial objects. (left column) Permutations, subsets, trees, and graphs. (right column) Point sets, polygons, and strings

An algorithmic problem is specified by describing the complete set of instances it must work on and of its output after running on one of these instances.

There is a distinction between a general problem and an instance of a problem. E.g:

Problem: Sorting
Input: A sequence of n keys a1,...,an.
Output: The permutation (reordering) of the input sequence such that: a′1 ≤ a′2 ≤ ··· ≤ a′n−1 ≤ a′n.
Instance of sorting problem: { Mike, Bob, Sally}; { 154, 245 }

An algorithm is a procedure that takes any of the possible input instances and transforms it to the desired output.

Three desirable properties for a good algorithm:

  1. Insertion sort is an algorithm to the sorting problem: English description:

    Start with a single element (thus forming a trivially sorted list) and then incrementally inserts the remaining elements so that the list stays sorted.

    Pseudocode:

    array = input sequence
    n = array size
    i = 0
    
    while i < n
       j = i
       while j > 0 AND array[j] < array[j - 1]
           swap element at `j` with element at `j - 1` in array
           decrement j by 1
       increment i by 1
    

    Code:

An animation of the logical flow of this algorithm on a particular instance (the letters in the word “INSERTIONSORT”)

Robot Tour Optimization

Problem: Robot Tour Optimization (aka: Traveling Salesman Problem [TSP]).
Input: A set S of n points in the plane.
Output: What is the shortest cycle tour that visits each point in the set S?

Nearest-neighbor heuristic

A good instance for the nearest-neighbor heuristic

  1. English Description:

    Starting from some point p0, we walk first to its nearest neighbor p1 From p1, we walk to its nearest unvisited neighbor. Repeat this process until we run out of unvisited points After which we return to p0 to close off the tour.

  2. Pseudocode:
    NearestNeighbor(P)
       Pick and visit an initial point p₀ from P
       p = p₀
       i = 0
       
       While there are still unvisited points
           i = i + 1
           Select pᵢ to be the closest unvisited point to pᵢ₋₁
           Visit pᵢ
       Return to p₀ from pₙ₋₁
    
  3. Pros:
  4. Cons: It's wrong — It always finds a tour, but it doesn’t necessarily find the shortest possible tour. E.g: A bad instance for the nearest-neighbor heuristic (top) & the optimal solution (bottom):

Closest-pair heuristic

  1. English description:

    Repeatedly connect the closest pair of endpoints whose connection will not create a problem, such as premature termination of the cycle. Each vertex begins as its own single vertex chain. After merging everything together, we will end up with a single chain containing all the points in it. Connecting the final two endpoints gives us a cycle. At any step during the execution of this closest-pair heuristic, we will have a set of single vertices and vertex-disjoint chains available to merge.

  2. Pseudocode:

Reasoning about Correctness

Correct algorithms usually come with a proof of correctness, which is an explanation of why we know that the algorithm must take every instance of the problem to the desired result.

  1. We need tools to distinguish correct algorithms from incorrect ones, the primary one of which is called a proof.
  2. A proper mathematical proof consists of several parts:
  3. A proof is indeed a demonstration. Proofs are useful only when they are honest, crisp arguments that explain why an algorithm satisfies a non-trivial correctness property.

Problems and Properties

  1. It is impossible to prove the correctness of an algorithm for a fuzzily-stated problem.
  2. Problem specifications have two parts:
  3. An important technique in algorithm design is to narrow the set of allowable instances until there is a correct and efficient algorithm. For example, we can restrict a graph problem from general graphs down to trees, or a geometric problem from two dimensions down to one.
  4. There are two common traps when specifying the output requirements of a problem:

Expressing Algorithms

  1. The heart of any algorithm is an idea. If your idea is not clearly revealed when you express an algorithm, then you are using too low-level a notation to describe it.
  2. The three most common forms of algorithmic notation are
  3. All three methods are useful because there is a natural tradeoff between greater ease of expression and precision.

Demonstrating Incorrectness

  1. The best way to prove that an algorithm is incorrect is to produce a counter example, i.e: an instance on which it yields an incorrect answer.
  2. Good counterexamples have two important properties:

Induction and Recursion

  1. Recursion is mathematical induction in action. In both, we have general and boundary conditions, with the general condition breaking the problem into smaller and smaller pieces. The initial or boundary condition terminates the recursion.
  2. The simplest and most common form of mathematical induction infers that a statement involving a natural number n (that is, an integer n ≥ 0) holds for all values of n. The proof consists of two steps:
  3. The hypothesis in the inductive step, that the statement holds for n = k, is called the induction/inductive hypothesis. You're doing a thought experiment of what would happen if it was true for n = k. It might be clearer to use the phrase "suppose true when n = k" rather than "assume true when n = k" to emphasise this.
  4. To prove the inductive step, one assumes the induction hypothesis for n = k and then uses this assumption to prove that the statement holds for n = k + 1. We try to manipulate the statement for n = k + 1 so that it involves the case for n = k (which we assumed to be true).

Inductive proof for Insertion sort

  1. The basis case consists of a single element, and by definition a one-element array is completely sorted.
  2. We assume that the first n - 1 elements of array A are completely sorted after n - 1 iterations of insertion sort.
  3. To insert one last element x to A, we find where it goes, namely the unique spot between the biggest element less than or equal to x and the smallest element greater than x. This is done by moving all the greater elements back by one position, creating room for x in the desired location. ■

Modeling the Problem

  1. Modeling is the art of formulating your application in terms of precisely described, well-understood problems.
  2. Proper modeling is the key to applying algorithmic design techniques to real-world problems — it can eliminate the need to design an algorithm, by relating your application to what has been done before.
  3. Real-world applications involve real-world objects — like a system to route traffic in a network or find the best way to schedule classrooms in a university.
  4. Most algorithms, however, are designed to work on rigorously defined abstract structures such as permutations, graphs, and sets.
  5. To exploit the algorithms literature, you must learn to describe your problem abstractly, in terms of procedures on such fundamental structures.
  6. The act of modeling reduces your application to one of a small number of existing problems and structures. Such a process is inherently constraining, and certain details might not fit easily into the given target problem.

Proof by Contradiction

  1. The basic scheme of a contradiction argument is as follows:
  2. The classic contradiction argument is Euclid's proof that there are an infinite number of prime numbers:
  3. For a contradiction argument to be convincing, the final consequence must be clearly, ridiculously false.

Estimation

  1. Principled guessing is called estimation.
  2. Estimation problems are best solved through some kind of logical reasoning process, typically a mix of principled calculations and analogies.
  3. Principled calculations give the answer as a function of quantities that either you already know, can look up on Google, or feel confident enough to guess.
  4. Analogies reference your past experiences, recalling those that seem similar to some aspect of the problem at hand.

Exercises

Finding counter examples

  1. Show that a + b can be less than min(a, b).

    When:
       a and b < 0 (i.e: negative)
       a <= b
    Then:
       min(a, b) = a
       a + b < a
    Example:
       min(-6, -5) = -6
       -6 + -5 = -6 -5 = -11
       -11 < -6

  2. Show that a × b can be less than min(a, b).

    When:
       a < 0 (i.e: negative)
       b > 0 (i.e: positive)
    Then:
       min(a, b) = a
       a * b < a
    Example:
       min(-3, 4) = -3
       -3 * 4 = -12
       -12 < -3

  3. Design/draw a road network with two points a and b such that the fastest route between a and b is not the shortest route.

a──────c──────b
│             │
│             │
└────d────────┘

Route `acb` have a toll gate, is in development and have a speed limit.
Route `adb` is longer, but has none of these impediment.

Route `adb` will be faster than the shorter `acb` route.
  1. Design/draw a road network with two points a and b such that the shortest route between a and b is not the route with the fewest turns.
    a────┐   ┌────b
    │    └─c─┘    │
    │             │
    └──────d──────┘
    
    Route `acb` is the shortest but has 4 turns.
    Route `adb` is the longest and has only 2 turns.
    

Induction

  1. Prove that i=1ni=n(n+1)2\sum_{i = 1}^{n} i = \frac{n(n + 1)}{2} for n ≥ 0, by induction.

Chapter 2: Algorithm Analysis

The analysis of algorithms is the process of finding the computational complexity of algorithms.

For an ordered list of size nn ($33$ in the example above), binary search takes at most log2n\log_2 n check steps ($5$ steps in the example above); while linear search takes at most nn check steps ($28$ steps in the example above).

🏋️‍♀️ Computational complexity

The computational complexity or simply complexity of an algorithm is the amount of resources required to run it.

Common types of resources include:

  1. Time — Time complexity is the computational complexity that describes the amount of computer time it takes to run an algorithm. When "complexity" is used without qualification, this generally means time complexity.
  2. 🫙 Memory — Space complexity is the computational complexity that describes the amount of memory it takes to run an algorithm.
  3. 🎙️ Communication — The necessary amount of communication between executing parties in a distributed algorithm.

The computational complexity of an algorithm can be measured only when given a model of computation.

💻 The RAM model of computation

The most commonly used model of computation is the Random Access Machine (RAM) which is a hypothetical computation model where:

The RAM is a simple model of how computers perform. It allows the analysis of algorithms in a machine-independent way by assuming that simple operations take a constant amount of time on a given computer and change only by a constant factor when run on a different computer.

It doesn’t capture the full complexity of real computers (e.g. multiplying two numbers takes more time than adding two numbers on most processors). However, the RAM is still an excellent model for understanding how an algorithm will perform on a real computer.

A model is a simplification or approximation of reality and hence will not reflect all of reality. […] "all models are wrong, but some are useful." — Model Selection and Multimodel Inference

🏭 Computational complexity functions

Under the RAM model, we measure the computational complexity by counting the number of simple operations (also called steps) an algorithm takes on a given problem instance.

As the amount of resources required to run an algorithm generally varies with the size of the input, the complexity is typically expressed as a function nf(n)n \rightarrow f(n), where nn is the size of the input and f(n)f(n) is the computational complexity.

Aside: We can also use any other characteristic of the input influencing the computational complexity.

However, the complexity of an algorithm may vary dramatically for different inputs of the same size. Therefore, three complexity functions are commonly used:

  1. 🔵 Worst-case complexity: is the maximum of the complexity over all inputs of size nn.
  2. 🟢 Average-case complexity: is the average of the complexity over all inputs of size nn.
  3. 🔴 Best-case complexity: is the minimum of the complexity over all inputs of size nn.
Line graph of worst, average, best case complexity

📈 Big oh notation

There are three problems with the complexity functions above:

  1. The exact computational complexity functions for any algorithm are liable to be complicated because they tend to have lots of up-and-down bumps.
  2. In addition, these exact values provide little practical application, as any change of computer or model of computation would change the complexity somewhat.
  3. Moreover, the resource use is not critical for small values of nn.

Complexity theory seeks to quantify the intrinsic time requirements of algorithms (independent of any external factors), that is, the basic time constraints an algorithm would place on any computer .

For these reasons, one generally focuses on the behavior of the complexity for large nn, that is on its asymptotic behavior when nn tends to infinity. Therefore, the complexity is generally expressed by using big O notation.

Big O notation is a mathematical notation that describes the limiting behavior of a function when the argument tends towards infinity. The formal definition of the big O notations are:

1. Big O (Upper bound)

Notation: f(n)=O(g(n))f(n) = O(g(n))

Meaning: There exists some constant cc such that f(n)cg(x)|f(n)| \leq c \cdot |g(x)| for large enough nn (i.e. for all nn0n \geq n_0, for some constant $n_0$).

Big O graph

2. Big Omega (Lower bound)

Notation: f(n)=Ω(g(n))f(n) = \Omega(g(n))

Meaning: Means there exists some constant cc such that f(n)cg(x)|f(n)| \geq c \cdot |g(x)| for large enough nn (i.e. for all nn0n \geq n_0, for some constant $n_0$).

Big Omega graph

3. Big Theta (Tight bound [upper & lower bound])

Notation: f(n)=Θ(g(n))f(n) = \Theta(g(n))

Meaning: Means there exists some constants c1c_1 and c2c_2 such that f(n)c1g(x)|f(n)| \leq c_1 \cdot |g(x)| and f(n)c2g(x)|f(n)| \geq c_2 \cdot |g(x)| for large enough nn (i.e. for all nn0n \geq n_0, for some constant $n_0$).

Big Theta graph

🙅‍♂️ Misconception: Conflating Big-O with complexity functions

The best/average/worst-case computational complexity is a function of the size of the input for a certain algorithm.

The OO, Ω\Omega, and Θ\Theta notations are used to describe the relationship between two functions.

Hence, each best/average/worst case computational complexity function has its corresponding OO, Ω\Omega, and Θ\Theta notation.

Conflating the big O notations with the computational complexity functions likely stems from the fact that in everyday use, “the big O of the worst-case computational complexity function” is used interchangeably with just “big O”, “time complexity”, “complexity”, etc.

🙈 Abuse of notation

There are two accepted abuses of notation in the computer science industry:

  1. Abuse of the equal sign: Technically, the appropriate notation is f(n)O(g(n))f(n) \in O(g(n)), because the equal sign does not imply symmetry.

  2. Abuse of the Big O notation: The big O notation is often used to describe an asymptotic tight bound where using big Theta Θ\Theta notation would be more factually appropriate. Big O notation provides an upper bound, but it doesn't necessarily provide a tight upper bound. For example, 2n2+3n+12n^2 + 3n + 1 is O(n2)O(n^2), but it is also O(n3)O(n^3), O(n4)O(n^4), etc. For contrast, the equation is only Θ(n2)\Theta(n^2).

Common Big O functions

The function g(x)g(x) appearing within the O()O(·) is typically chosen to be as simple as possible, omitting constant factors and lower-order terms.

Common Big O function classes

Arithmetic operations with Big O notation

  1. Addition:

  2. Multiplication by a constant:

  3. Multiplication by non-constant:

  4. Transitivity:

Reasoning about an algorithm's efficiency

1. Selection sort

// Selection sort repeatedly identifies the smallest remaining unsorted element
// and puts it at the end of the sorted portion of the array.
void selectionSort(int arr[], int n) {
    int i, j, minimum_element_index;

    // One by one move the boundary of the unsorted subarray
    for (i = 0; i < n; i++) {

        // Find the minimum element in the unsorted array
        minimum_element_index = i;  // Assume the current element is the minimum
        
        for (j = i + 1; j < n; j++) {
            // If we find a smaller element, update the minimum_element_index
            if (arr[j] < arr[minimum_element_index]) {
                minimum_element_index = j;
            }
        }

        // Swap the found minimum element with the first element
        swap(&arr[minimum_element_index], &arr[i]);
    }
}

Analysis of the selection sort algorithm:

  1. The outer loop goes around nn times.
  2. The nested inner loop goes around n(i+1)n - (i + 1) times, where ii is the index of the outer loop.
  3. The exact number of times the if statement is executed is:T(n)=i=0n1j=i+1n11T(n) = \sum_{i=0}^{n-1} \sum_{j=i+1}^{n-1} 1
  4. What the sum is doing is adding up the non-negative integers in decreasing order starting from n1n - 1:T(n)=(n1)+(n2)+(n3)+...+2+1+0T(n) = (n - 1) + (n - 2) + (n - 3) + ... + 2 + 1 + 0
  5. The formula for the sum of the first nn non-negative integers (derived from the sum of an arithmetic series) is given by:i=0n1i=n(n1)2\sum_{i=0}^{n-1} i = \frac{n(n - 1)}{2}
  6. Hence, we can simply the summation to:i=0n1ni1=n(n1)2=n2n2\sum_{i=0}^{n-1} n - i - 1 = \frac{n(n - 1)}{2}\\ = \frac{n^2 - n}{2}
  7. Therefore the time complexity of selection sort is:

Logarithms and their applications

A logarithm is an inverse exponential function:

bx=yx=logbyb(logby)=yb^x = y \\ x = \log_b y \\ b^(\log_b y) = y

Exponential functions grow at a fast rate while logarithmic functions grow at a slow rate. Logarithms arise in any process where things are repeatedly halved. e.g.

  1. Logarithms and binary search:

  2. Logarithms and trees:

  3. Logarithms and bits:

Product property of logarithms

logb(ac)=logb(a)+logb(c)\log_b (a \cdot c) = \log_b (a) + \log_b (c)

A direct consequence of the product property is the power property: $ \log_b (a^c) = c \cdot \log_b (a) $

Since cc is a constant, this implies raising the logarithm’s argument to a power doesn't affect the growth rate: $ \log_b (a^c) = \Theta (\log_b (a)) $

Change of base property of logarithms

Changing the base of logb(a)\log_b (a) to base-c simply involves multiplying by logc(b)\log_c (b): $ \log_b (a) = \frac{\log_c (a)}{\log_c (b)} \

\log_c (a) = \log_b (a) \cdot \log_c (b) $

Since cc and bb are constants, this property implies that the base of a logarithm has no impact on the growth rate: $ \log_2 (1,000,000) = 19.9316\ \log_3 (1,000,000) = 12.5754\ \log_100 (1,000,000) = 3 $

Aside: In practice, an asymptotically inefficient algorithm may be more efficient for small input sizes. This is particularly used in hybrid algorithms, like Timsort, which use an asymptotically efficient algorithm (here merge sort, with time complexity $n \log n$), but switch to an asymptotically inefficient algorithm (here insertion sort, with time complexity $n^2$) for small data, as the simpler algorithm is faster on small data.

Chapter 3: Data structures

  1. A data type (or simply type) is a collection of data values, usually specified by:
  2. An abstract data type (ADT) is a data type that does not specify the concrete representation of the data. They are defined by their behavior (semantics) from the point of view of a user of the data, specifically in terms of:
  3. The generic definition of data structure is anything that can hold your data in a structured way. ADT contrasts with data structures, which are concrete representations of data, and are the point of view of an implementer.
  4. The distinction between ADTs and data structures lies in the POV / level of abstraction. Some important points:
  5. Data structures can be classified into:

Arrays

  1. Arrays are data structures of fixed-size elements stored contiguously such that each element can be efficiently located by its index.ElementAddress(i)=FirstElementAddress+(iElementSize)ElementAddress(i) = FirstElementAddress + (i \cdot ElementSize)
  2. Advantages of arrays:
  3. The primary disadvantage of arrays is that the number of elements (i.e. the array size) is fixed. The capacity needs to be specified at allocation.
  4. Dynamic arrays overcome the fixed-size limitation of static arrays.
  5. It does so by internally resizing when its capacity is exceeded:
  6. To avoid incurring the cost of resizing many times, dynamic arrays resize by a large amount, such as doubling in size.
  7. Expanding the array by any constant proportion aa ensures that inserting nn elements takes O(n)lt;/span> time overall, meaning that each insertion takes **amortized** <span class="good">$O(1)lt;/span> time. As $n elements are added, the capacities form a geometric progression.
  8. The key idea of amortized analysis is to consider the worst-case cost of a sequence of operations on a data structure, rather than the worst-case individual cost of any particular operation.
  9. The aggregate method is one of the methods for performing amortized analysis. In this method, the total cost of performing a sequence of mm operations on the data structure is divided by the number of operations mm, yielding an average cost per operation, which is defined as the amortized cost.
  10. The aggregate method of amortized analysis applied to dynamic arrays with doubling:
  1. The key idea behind amortized analysis is that the cost of a particular operation can be partially paid for by the cost of other operations that are performed later. It avoids the limitations of worst-case analysis, which can overestimate the performance of a data structure if the worst-case scenario is unlikely to occur frequently.

Pointers and linked structures

Singly linked list

  1. Pointers represent the address of a location in memory. Pointers are the connections that hold the nodes (i.e. elements) of linked data structures together.

  2. In C:

  3. All linked data structures share these properties:

    Example linked-list displaying these properties:

     
    typedef struct list {
        data_type data; /* Data field */
        struct list *next; /* Pointer field */
    } list;
    
  4. The linked-list is the simplest linked structure.

  5. Singly linked-list has a pointer to only the successor whereas a doubly linked-list has a pointer to both the predecessor and successor.

  6. Searching for data x in a linked-list recursively:

     
    list *search_list(list *my_list, data_type x) {
        if (my_list == NULL) {
            return(NULL);
        }
    
        // If `x` is in the list, it's either the first element or located in the rest of the list.
        if (my_list->data == x) {
            return(my_list);
        } else {
            return(search_list(my_list.next, x));
        }
    }
    
  7. Inserting into a singly linked-list at the head:

     
    list *insert_list(list **my_list, data_type x) {
       list *p; /* temporary pointer */
       p = malloc(sizeof(list));
       p->data = x;
       p->next = *my_list;
       // `**my_list` denotes that `my_list` is a pointer to a pointer to a list node.
       // This line copies `p` to the place pointed to by `my_list`,
       // which is the external variable maintaining access to the head of the list.
       *my_list = p;
    }
    
  8. Deletion from a list:

     // Used to find the predecessor of the item to be deleted.
    list *item_ahead(list *my_list, list *x) {
       if ((my_list == NULL) || (my_list->next == NULL) {
           return(NULL);
       }
    
       if ((my_list->next) == x) {
           return(my_list);
       } else {
           return(item_ahead(my_list->next, x));
       }
    }
    
    // This is called only if `x` exists in the list.
    void *delete_list(list **my_list, list **x) {
       list *p; /* element pointer */
       list *pred; /* predecessor pointer */
    
       p = *my_list;
       pred = item_ahead(*my_list, *x);
    
       // Given that we assume `x` exists in the list,
       // `pred` is only null when the first element is the target.
       if (pred == NULL) { /* splice out of list */
          // Special case: resets the pointer to the head of the list
          // when the first element is deleted.
          *my_list = p->next;
       } else {
          pred->next = (*x)->next
       }
    
       free(*x) /* free memory used by node */
    }
    
  9. The advantages of linked structures over static arrays include:

  10. Both arrays and lists can be thought of as recursive objects:

  11. An array is only recursive conceptually: Every array "contains" a sub-array. However, an array isn't recursive in code: A structure is recursive if while navigating through it, you come across "smaller" versions of the whole structure. While navigating through an array, you only come across the elements that the array holds, not smaller arrays.

Stacks

  1. Stacks are an ADT that supports retrieval by last-in, first-out (LIFO).
  2. Primary operations are:

Queues

Queue

  1. Queues are an ADT that supports retrieval by first-in, first-out (FIFO).
  2. Primary operations are:
  3. Stacks and queues can be effectively implemented using arrays or linked-list.

Dictionaries

  1. A dictionary is an ADT that stores a collection of (key, value) pairs, such that each possible key appears at most once in the collection.
  2. The primary operations of a dictionary are:
  3. These are two simple but less common implementations of a dictionary:
  4. These are the two common data structures used to implement a dictionary:
  5. Binary search tree based maps are in-order and hence can satisfy range queries (i.e. find all values between two bounds) whereas a hash-map can only efficiently find exact values.

Binary search tree (BST)

  1. A binary tree is a tree data structure in which each node has at most two children, referred to as the left child and the right child.
  2. A binary tree can be viewed as a linked-list with two pointers per node.
  3. A rooted tree is a tree in which one node has been designated the root.
  4. A rooted binary tree is recursively defined as either being:
  5. A binary search tree is a rooted binary tree data structure such that for any node xx, all nodes in the left subtree of xx have keys<xkeys < x while all nodes in the right subtree of xx have keys>xkeys > x.
  6. Binary search trees’ height range from log2n\log_2 n (when balanced) to nn (when degenerate).
  7. The time complexity of operations on the binary search tree is linear with respect to the height of the tree O(h)O(h).
  8. Hence, in a balanced binary search tree, the nodes are laid out so that each comparison skips about half of the remaining tree, the lookup performance is proportional to that of the binary logarithm $O(\log n)$.
  9. An implementation of a binary tree:
     
    typedef struct tree {
        data_type data; /* Data field */
        struct tree *parent; /* Pointer to parent */
        struct tree *left; /* Pointer to left chid */
        struct tree *right; /* Pointer to right child */
    } tree;
    
  10. Searching in a binary search tree:

Priority queue

  1. A priority queue is an ADT similar to a regular queue or stack ADT. Each element in a priority queue has an associated priority.
  2. Depending on the implementation, a priority queue can serve minimum priority elements first or maximum priority elements first. You can convert one implementation into the other by negating the priority on each element addition.
  3. Priority queues are often implemented using heaps.
  4. A priority queue can also be inefficiently implemented as an unsorted list or a sorted list.
  5. A priority queue can be used for sorting: insert all the elements to be sorted into a priority queue, and sequentially remove them.
  6. The primary operations of a priority queue are:
  7. Stacks and queues can be implemented as particular kinds of priority queues, with the priority determined by the order in which the elements are inserted.

Hash tables

Hash table

  1. Hash tables are a data structure that efficiently implements a dictionary. They exploit the fact that looking an element up in an array takes constant time once you have its index.
  2. The basic idea is to pick a hash function hh that maps every possible key xx to a small integer h(x)h(x). Then we store xx and its value in an array at index $h(x)$; the array itself is essentially the hash table.
  3. A hash function hh maps the universe UU of keys to array indices within the hash table.h:U{0,,n1}h : U → \{ 0, …, n - 1 \}
  4. Properties of a good hash function:
  5. A hash function for an arbitrary key xx (like a string) is typically defined as:h(x)=toInteger(x)modnh(x) = toInteger(x) \bmod n
  6. This is the polynomial function that Java uses to convert strings to integers:s[0]31n1+s[1]31n2+...+s[n1]s[0]*31^{n-1} + s[1]*31^{n-2} + ... + s[n-1] Which translates into the following code:
    int hashcode = 0;
    for (int i = 0; i < s.length(); i++) {
        hashcode = (hashcode * 31) + s.charAt(i);
    }
    
  7. A collision occurs when:h(j)=h(k)jkh(j) = h(k) \land j \neq k
  8. Collisions can be minimized but cannot be eliminated (see Pigeon hole principle). It’s impossible to eliminate collisions without knowing the UU ahead of time.
  9. The two common methods for collision resolution are separate chaining and open addressing.
  10. In separate chaining, the values of the hash-table’s array is a linked-list.
  11. In open addressing, every key and its value is stored in the hash-table’s array itself, and the resolution is performed through probing.
  12. Search algorithms that use hashing consist of two separate parts: hashing and collision resolution.
  13. Other uses of hashing (or a hash table):
  14. Time complexity in big O notation

| Operation | Average | Worst case | |-----------|---------|------------| | Search | Θ(1)Θ(1) | $O(n)$ | | Insert | Θ(1)Θ(1) | $O(n)$| | Delete | Θ(1)Θ(1) | $O(n)$ | 15. Space complexity is $O(n)$.

Excercises

  1. A common problem for compilers and text editors is determining whether the parentheses in a string are balanced and properly nested. For example, the string ((())())() contains properly nested pairs of parentheses, while the strings )()( and ()) do not. Give an algorithm that returns true if a string contains properly nested and balanced parentheses, and false if otherwise. For full credit, identify the position of the first offending parenthesis if the string is not properly nested and balanced.

    Solution
    fun test() {
        System.out.println(areParentheseProperlyBalanced("((())())()"))
        System.out.println(areParentheseProperlyBalanced(")()("))
        System.out.println(areParentheseProperlyBalanced("())"))
        System.out.println(areParentheseProperlyBalanced(")))"))
        System.out.println(areParentheseProperlyBalanced("(("))
     }
    
    /**
      * @return -1 if [string] is valid, else a positive integer
      * that provides the position of the offending index.
      */
    fun areParentheseProperlyBalanced(string: String): Int {
        val stack = Stack<Pair<Char, Int>>()
    
        string.forEachIndexed { index, char ->
             if (char == '(') {
                 stack.push(char to index)
             } else if (char == ')') {
                 if (stack.empty()) {
                     return index
                 }   
                 stack.pop()
             } else {
                 throw IllegalArgumentException("Only parenthesis are supported")
             }
         }   
        
        return if (stack.empty()) -1 else stack.peek().second
     }
    
  2. Give an algorithm that takes a string SS consisting of opening and closing parentheses, say )()(())()()))())))(, and finds the length of the longest balanced parentheses in SS, which is 12 in the example above. (Hint: The solution is not necessarily a contiguous run of parenthesis from $S$)

    Solution
    fun test() {
        System.out.println(lengthOfLongestBalancedParentheses("((())())()"))
        System.out.println(lengthOfLongestBalancedParentheses(")()(())()()))())))("))
        System.out.println(lengthOfLongestBalancedParentheses(")()("))
        System.out.println(lengthOfLongestBalancedParentheses("())"))
        System.out.println(lengthOfLongestBalancedParentheses(")))"))
        System.out.println(lengthOfLongestBalancedParentheses("(("))
    }
    
    fun lengthOfLongestBalancedParentheses(string: String): Int {
        val stack = Stack<Char>()
        var numBalancedParenthesis = 0
    
        string.forEachIndexed { index, char ->
            if (char == '(') {
                stack.push(char)
            } else if (char == ')') {
                if (!stack.empty()) {
                    stack.pop()
                    numBalancedParenthesis++
                }
            }
        }
    
        // Multiplied by 2 because each balanced pair has a length of 2.
        return numBalancedParenthesis * 2
    }
    
  3. Give an algorithm to reverse the direction of a given singly linked list. In other words, after the reversal all pointers should now point backwards. Your algorithm should take linear time.

    Solution
    fun test() {
        val node1 = Node("Elvis", null)
        
        val node2 = Node("Chidera", null)
        node1.next = node2
        
        val node3 = Node("Nnajiofor", null)
        node2.next = node3
        
        System.out.println(node1)
        System.out.println(reverse(node1))
    }
    
    data class Node(
        val element: String,
        var next: Node?
    )
    
    /**
     * Work through an example:
     * reverse(Node1 -> Node2 -> Node3)
     * 
     * Node1 level:
     * val last = reverse(Node2 -> Node3) 🛑
     * 
     * Node2 level:
     * val last = reverse(Node3) 🛑
     * 
     * Node3 level:
     * return Node3
     * 
     * Back to Node2 level:
     * val last = Node3 🟢
     * Node3.next = Node2
     * Node2.next = null
     * return last
     * 
     * Back to Node1 level:
     * val last = Node3 🟢
     * Node2.next = Node1
     * Node1.next = null
     * return last
    */
    fun reverse(node: Node): Node {
        // Base case
        if (node.next == null) return node
        
        val last = reverse(node.next!!)
        node.next!!.next = node
        node.next = null
        return last
    }
    
  4. Design a stack SS that supports S.push(x), S.pop(), and S.findmin(), which returns the minimum element of SS. All operations should run in constant time.

    Solution
    fun test() {
        val stack = Stack()
        stack.push(50)
        stack.push(40)
        stack.push(30)
        stack.push(20)
        stack.push(10)
        
        System.out.println(stack.findMin())
        stack.pop()
        System.out.println(stack.findMin())
        stack.pop()
        System.out.println(stack.findMin())
        stack.pop()
        System.out.println(stack.findMin())
        stack.pop()
        System.out.println(stack.findMin())
        stack.pop()
    }
    
    data class Element(
        val num: Int,
        internal val minNumSoFar: Int
    )
    
    class Stack {
        
        private val list = mutableListOf<Element>()
        
        fun push(num: Int) {
            list += Element(
                num = num,
                minNumSoFar = Math.min(num, list.lastOrNull()?.minNumSoFar ?: num)
            )    
        }
        
        fun pop(): Int {
            return list.removeLast().num
        }
        
        fun findMin(): Int {
            return list.last().minNumSoFar
        }
    }
    
  5. We have seen how dynamic arrays enable arrays to grow while still achieving constant-time amortized performance. This problem concerns extending dynamic arrays to let them both grow and shrink on demand.

    a. Consider an underflow strategy that cuts the array size in half whenever the array falls below half full. Give an example sequence of insertions and deletions where this strategy gives a bad amortized cost.

    b. Then, give a better underflow strategy than that suggested above, one that achieves constant amortized cost per deletion.

    Solution

    a. A sequence of insert, insert, delete, insert, delete, insert will lead to a bad amortized cost because the array will be busy resizing for most of the operations.

    b. A better underflow strategy is to cut the array size in half whenever the array falls below 1/41/4. 1/41/4 is arbitrary, but it gives the array good "slack" space. The goal is to select a number such that the array is not busy resizing for most of the operation. The smaller the cut-off ratio, the smaller the number of resizing but the more space the array uses.

  6. Suppose you seek to maintain the contents of a refrigerator so as to minimize food spoilage. What data structure should you use, and how should you use it?

    Solution

    Use a priority queue. The expiry date is the priority of each element. Elements with the lowest expiry date are served first (minElement).

  7. Work out the details of supporting constant-time deletion from a singly linked list as per the footnote from page 79, ideally to an actual implementation. Support the other operations as efficiently as possible.

    Solution
    fun test() {
        val node3 = Node(3)
        val node2 = Node(2, node3)
        val node1 = Node(1, node2)
    
        println(node1)
        println(deleteInConstantTime(node1, node2))
        println(deleteInConstantTime(node1, node3))
        println(deleteInConstantTime(node1, node1))
    }
    
    fun deleteInConstantTime(head: Node, search: Node): Node? {
        if (head == search) {
            // Handle head case
            return head.next
        }
    
        var node: Node? = head
        var prevNode: Node? = null
        while (node != null) {
            if (node == search) {
                if (node.next == null) {
                    // Handle tail case
                    prevNode?.next = null
                } else {
                    node.element = node.next!!.element
                    node.next = node.next?.next
                }
    
                break
            } else {
                prevNode = node
                node = node.next
            }
        }
    
        return head
    }
    
    data class Node(
        var element: Int,
        var next: Node? = null
    )
    
  8. Tic-tac-toe is a game played on an nnn * n board (typically n = 3$) where two players take consecutive turns placing “O” and “X” marks onto the board cells. The game is won if $n consecutive “O” or “X” marks are placed in a row, column, or diagonal. Create a data structure with O(n)O(n) space that accepts a sequence of moves, and reports in constant time whether the last move won the game.

    Solution
    fun test() {
        var tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playX(1, 1))
        assertFalse(tickTacToe.playO(1, 2))
        assertFalse(tickTacToe.playX(1, 3))
    
        tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playO(2, 1))
        assertFalse(tickTacToe.playX(2, 2))
        assertFalse(tickTacToe.playO(2, 3))
    
        tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playX(3, 1))
        assertFalse(tickTacToe.playX(3, 2))
        assertTrue(tickTacToe.playX(3, 3))
    
        tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playX(1, 1))
        assertFalse(tickTacToe.playO(2, 1))
        assertFalse(tickTacToe.playX(3, 1))
    
        tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playO(1, 2))
        assertFalse(tickTacToe.playX(2, 2))
        assertFalse(tickTacToe.playO(3, 2))
    
        tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playX(1, 3))
        assertFalse(tickTacToe.playX(2, 3))
        assertTrue(tickTacToe.playX(3, 3))
    
        tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playO(1, 1))
        assertFalse(tickTacToe.playO(2, 2))
        assertTrue(tickTacToe.playO(3, 3))
    
        tickTacToe = TickTacToe(3)
        assertFalse(tickTacToe.playO(1, 3))
        assertFalse(tickTacToe.playO(2, 2))
        assertTrue(tickTacToe.playO(3, 1))
    }
    
    private fun assertTrue(value: Boolean) {
        require(value)
        println("Won!!!")
    }
    
    private fun assertFalse(value: Boolean) {
        require(!value)
        println("Not won yet")
    }
    
    class TickTacToe(private val n: Int) {
    
        private val columns = Array(n) {
            Slot(n)
        }
    
        private val rows = Array(n) {
            Slot(n)
        }
    
        private val diagonal = Slot(n)
    
        private val antiDiagonal = Slot(n)
    
        fun playX(rowPosition: Int, columnPosition: Int): Boolean {
            return play('X', rowPosition, columnPosition)
        }
    
        fun playO(rowPosition: Int, columnPosition: Int): Boolean {
            return play('O', rowPosition, columnPosition)
        }
    
        private fun play(char: Char, rowPosition: Int, columnPosition: Int): Boolean {
            return rows[rowPosition.toIndex].play(char) ||
                    columns[columnPosition.toIndex].play(char) ||
                    (rowPosition == columnPosition && diagonal.play(char)) ||
                    ((rowPosition + columnPosition) == (n + 1) && antiDiagonal.play(char))
        }
    
        private val Int.toIndex get() = this - 1
    
        class Slot(private val n: Int) {
    
            private var number = 0
    
            fun play(char: Char): Boolean {
                val increment = if (char == 'X') 1 else -1
                val target = if (char == 'X') n else -n
    
                number += increment
                return number == target
            }
        }
    }
    
  9. Write a function which, given a sequence of digits 2–9 and a dictionary of nn words, reports all words described by this sequence when typed in on a standard telephone keypad. For the sequence 269 you should return any, box, boy, and cow, among other words.

    Solution
    fun test() {
        println(words(arrayOf(2, 6, 9)))
        println(words(arrayOf(7, 6, 7, 7)))
    }
    
    fun words(inputDigits: Array<Int>): List<String> {
        val words = setOf(
            "pops",
            "any",
            "box",
            "boy",
            "cow",
            "dad",
            "mom",
        )
        val charToDigitMapping = mapOf(
            'a' to 2,
            'b' to 2,
            'c' to 2,
            'd' to 3,
            'e' to 3,
            'f' to 3,
            'g' to 4,
            'h' to 4,
            'i' to 4,
            'j' to 5,
            'k' to 5,
            'l' to 5,
            'm' to 6,
            'n' to 6,
            'o' to 6,
            'p' to 7,
            'q' to 7,
            'r' to 7,
            's' to 7,
            't' to 8,
            'u' to 8,
            'v' to 8,
            'w' to 9,
            'x' to 9,
            'y' to 9,
            'z' to 9,
        )
    
        val matchingWords = mutableListOf<String>()
        words.forEach { word ->
            word.forEachIndexed { index, char ->
                val charDigit = charToDigitMapping[char] ?: return@forEach
                val inputDigitAtIndex = inputDigits.getOrNull(index) ?: return@forEach
                if (charDigit != inputDigitAtIndex) {
                    return@forEach
                }
            }
    
            matchingWords += word
        }
    
        return matchingWords
    }
    
  10. Two strings XX and YY are anagrams if the letters of XX can be rearranged to form YY. For example, silent/listen, and incest/insect are anagrams. Give an efficient algorithm to determine whether strings XX and YY are anagrams.

    Solution
    fun test() {
        println(isAnagram("silent", "listen"))
        println(isAnagram("silence", "listen"))
        println(isAnagram("incest", "insect"))
        println(isAnagram("incest", "insects"))
    }
    
    fun isAnagram(s1: String, s2: String): Boolean {
        val map = mutableMapOf<Char, Int>()
        
        s1.forEach { char ->
            map[char] = map.getOrPut(char) { 0 } + 1
        }
        
        s2.forEach { char ->
            if (map.containsKey(char)) {
                map[char] = map.getValue(char) - 1
            } else {
                return false
            }
        }
        
        map.values.forEach { number ->
            if (number != 0) {
                return false
            }
        }
        
        return true
    }
    
  11. Design a dictionary data structure in which search, insertion, and deletion can all be processed in O(1)O(1) time in the worst case. You may assume the set elements are integers drawn from a finite set 1,2,...,n1, 2, ..., n and initialization can take O(N)O(N) time.

    Solution
    fun test() {
        val dictionary = Dictionary(10)
        dictionary.insert(4)
        println(dictionary.search(4))
        dictionary.delete(4)
        println(dictionary.search(4))
    }
    
    class Dictionary(val capacity: Int) {
        
        private val array = Array<Int?>(capacity) { null }
        
        fun search(element: Int): Int? {
            return array[element.toIndex]
        }
    
        fun delete(element: Int) {
            array[element.toIndex] = null
        }
    
        fun insert(element: Int) {
            array[element.toIndex] = element
        }
        
        private val Int.toIndex get() = this - 1
    }
    
  12. The maximum depth of a binary tree is the number of nodes on the path from the root down to the most distant leaf node. Give an O(n)O(n) algorithm to find the maximum depth of a binary tree with nn nodes.

    Solution
    fun test() {
        val root = BNode(
            element = 5,
            left = BNode(
                element = 3,
                left = BNode(
                    element = 2,
                    left = BNode(
                        element = 1,
                        left = null,
                        right = null
                    ),
                    right = null
                ),
                right = null
            ),
            right = BNode(
                element = 7,
                left = null,
                right = BNode(
                    element = 8,
                    left = null,
                    right = null
                )
            )
        )
    
        println(maxDepth(root))
    }
    
    fun maxDepth(root: BNode?): Int {
        if (root == null) {
            return 0
        }
    
        val leftDepth = maxDepth(root.left)
        val rightDepth = maxDepth(root.right)
    
        return max(leftDepth, rightDepth) + 1
    }
    
  13. Two elements of a binary search tree have been swapped by mistake. Give an O(n)O(n) algorithm to identify these two elements so they can be swapped back.

    Solution
    fun test() {
        val root = BNode(
            element = 5,
            left = BNode(
                element = 3,
                left = BNode(
                    element = 8, // Swapped
                    left = null,
                    right = null
                ),
                right = null
            ),
            right = BNode(
                element = 7,
                left = null,
                right = BNode(
                    element = 2, // Swapped
                    left = null,
                    right = null
                )
            )
        )
    
        println(root)
        val recoverer = SwappedNodeRecoverer()
        recoverer.recover(root)
        println(root)
    }
    
    class SwappedNodeRecoverer {
    
        private var prev: BNode? = null
        private var first: BNode? = null
        private var second: BNode? = null
    
        fun recover(root: BNode?) {
            recoverInternal(root)
    
            if (first != null) {
                val firstElement = first!!.element
                first!!.element = second!!.element
                second!!.element = firstElement
            }
        }
    
        private fun recoverInternal(root: BNode?) {
            if (root == null) {
                return
            }
    
            recoverInternal(root.left)
    
            if (prev != null && root.element < prev!!.element) {
                if (first == null) {
                    // This handles adjacent node case
                    first = prev
                    second = root
                } else {
                    second = root
                }
            }
    
            prev = root
    
            recoverInternal(root.right)
        }
    }
    
    data class BNode(
        var element: Int,
        var left: BNode? = null,
        var right: BNode? = null,
    )
    
  14. Given two binary search trees, merge them into a doubly linked list in sorted order.

    Solution
    fun test() {
        val s1 = BNode(
            element = 5,
            left = BNode(
                element = 3,
                left = null,
                right = null
            ),
            right = BNode(
                element = 7,
                left = null,
                right = null
            )
        )
    
        val s2 = BNode(
            element = 10,
            left = BNode(
                element = 8,
                left = null,
                right = null
            ),
            right = BNode(
                element = 12,
                left = null,
                right = null
            )
        )
    
        println(merge(s1, s2))
    }
    
    fun merge(tree1: BNode, tree2: BNode): Node {
        val tree1Nodes = toList(tree1)
        val tree2Nodes = toList(tree2)
    
        var i1 = 0
        var i2 = 0
        val list = Node(Integer.MIN_VALUE) // Sentinel
        var lastListNode = list
        val addListNode = { node: Node ->
            lastListNode.next = node
            lastListNode = node
        }
    
        while (i1 < tree1Nodes.size || i2 < tree2Nodes.size) {
            val tree1Node = tree1Nodes.getOrNull(i1)
            val tree2Node = tree2Nodes.getOrNull(i2)
    
            if (tree1Node == null && tree2Node != null) {
                addListNode(Node(tree2Node.element))
                i2++
            } else if (tree2Node == null && tree1Node != null) {
                addListNode(Node(tree1Node.element))
                i1++
            } else if (tree1Node!!.element < tree2Node!!.element) {
                addListNode(Node(tree1Node.element))
                i1++
            } else if(tree1Node.element > tree2Node.element) {
                addListNode(Node(tree2Node.element))
                i2++
            } else {
                addListNode(Node(tree1Node.element))
                i1++
                addListNode(Node(tree2Node.element))
                i2++
            }
        }
    
        return list.next!!
    }
    
    fun toList(tree: BNode?): MutableList<BNode> {
        if (tree == null) {
            return mutableListOf()
        }
    
        return (toList(tree.left) + tree + toList(tree.right)).toMutableList()
    }
    
    data class BNode(
        val element: Int,
        var left: BNode?,
        var right: BNode?,
    )
    
    data class Node(
        val element: Int,
        var next: Node? = null
    )
    
  15. Describe an $O(n)$-time algorithm that takes an $n$-node binary search tree and constructs an equivalent height-balanced binary search tree. In a height-balanced binary search tree, the difference between the height of the left and right subtrees of every node is never more than 1.

    Solution
    fun test() {
        val root = BNode(
            element = 5,
            left = BNode(
                element = 3,
                left = BNode(
                    element = 2,
                    left = BNode(
                        element = 1,
                        left = null,
                        right = null
                    ),
                    right = null
                ),
                right = null
            ),
            right = BNode(
                element = 7,
                left = null,
                right = BNode(
                    element = 8,
                    left = null,
                    right = null
                )
            )
        )
    
        println("Unbalanced tree: $root")
        println("Balanced tree: " + toBalancedTree(root))
    }
    
    fun toBalancedTree(root: BNode): BNode {
        val nodes = toList(root)
        return sortedListToBinaryTree(nodes)!!
    }
    
    fun sortedListToBinaryTree(list: List<BNode>): BNode? {
        if (list.isEmpty()) {
            return null
        }
    
        val mid = list.size / 2
        val root = BNode(list[mid].element)
        root.left = sortedListToBinaryTree(list.slice(0 until mid))
        root.right = sortedListToBinaryTree(list.slice((mid + 1) until list.size))
    
        return root
    }
    
    fun toList(tree: BNode?): MutableList<BNode> {
        if (tree == null) {
            return mutableListOf()
        }
    
        return (toList(tree.left) + tree + toList(tree.right)).toMutableList()
    }
    
    data class BNode(
        val element: Int,
        var left: BNode? = null,
        var right: BNode? = null,
    )
    
  16. Find the storage efficiency ratio (the ratio of data space over total space) for each of the following binary tree implementations on nn nodes:

    Solution

    Storage efficiency ratio = /fracSpace used to store dataSpace used to store data and pointers/frac{Space \space used \space to \space store \space data}{Space \space used \space to \space store \space data \space and \space pointers} For option A: Space taken by a single node = (2 pointers * 4 bytes) + (1 pointer * 4 bytes) + (4 bytes) = 16 bytes Given nn nodes Storage efficiency ratio = 4n16n=14\frac{4n}{16n} = \frac{1}{4}

    For option B: Space taken by a single internal node = 2 pointers * 2 bytes = 4 bytes Space taken by a single leaf node = 4 bytes In a full tree, given nn leaf nodes, there are n1n-1 internal nodes. Storage efficiency ratio = 4n4n+4(n1)=4n4n+4n4=4n8n4=4n8n=1/2\frac{4n}{4n + 4(n-1)} = \frac{4n}{4n + 4n - 4} = \frac{4n}{8n - 4} = \frac{4n}{8n} = \frac{1}/{2} (As nn gets larger, the constant 44 doesn't matter, hence why it was dropped)

    Option B has a higher storage efficiency to option A.

  17. Give an O(n)O(n) algorithm that determines whether a given $n$-node binary tree is height-balanced (see Problem 3-15).

    Solution
    fun test() {
        val root = BNode(
            element = 5,
            left = BNode(
                element = 3,
                left = BNode(
                    element = 2,
                    left = BNode(
                        element = 1,
                        left = null,
                        right = null
                    ),
                    right = null
                ),
                right = null
            ),
            right = BNode(
                element = 7,
                left = null,
                right = BNode(
                    element = 8,
                    left = null,
                    right = null
                )
            )
        )
    
        println(isBalanced(root))
    }
    
    fun isBalanced(root: BNode?): Pair<Boolean, Int> {
        if (root == null) {
            return true to 0
        }
    
        val (isLeftBalanced, leftHeight) = isBalanced(root.left)
        if (!isLeftBalanced) {
            return false to 0
        }
    
        val (isRightBalanced, rightHeight) = isBalanced(root.right)
        if (!isRightBalanced) {
            return false to 0
        }
    
        if (abs(leftHeight - rightHeight) > 1) {
            return false to 0
        }
    
        return true to (max(leftHeight, rightHeight) + 1)
    }
    
    data class BNode(
        val element: Int,
        val left: BNode?,
        val right: BNode?,
    )
    
  18. Describe how to modify any balanced tree data structure such that search, insert, delete, minimum, and maximum still take O(logn)O(log n) time each, but successor and predecessor now take O(1)O(1) time each. Which operations have to be modified to support this?

    Solution

    TODO

  19. Suppose you have access to a balanced dictionary data structure that supports each of the operations search, insert, delete, minimum, maximum, successor, and predecessor in O(logn)O(log n) time. Explain how to modify the insert and delete operations so they still take O(logn)O(log n) but now minimum and maximum take O(1)O(1) time. (Hint: think in terms of using the abstract dictionary operations, instead of mucking about with pointers and the like.)

    Solution

    Use two variables to maintain the maximum and minimum element. On:

    • Insert(x): If x<minimumx < minimum, set xx as new minimum; If x>maximumx > maximum, set xx as new maximum.
    • Delete(x): If x=minimumx = minimum, set minimum = sucessor(x)$; If $x = maximum, set maximum=predecessor(x)maximum = predecessor(x).
  20. Design a data structure to support the following operations:

    All operations must take O(logn)O(log n) time on an $n$-element set.

    Solution

    g

  21. A concatenate operation takes two sets S1S_1 and S2S_2, where every key in S1S_1 is smaller than any key in S2S_2, and merges them. Give an algorithm to concatenate two binary search trees into one binary search tree. The worst-case running time should be O(h)O(h), where hh is the maximal height of the two trees.

    Solution
    fun test() {
        val s1 = BNode(
            element = 5,
            left = BNode(
                element = 3,
                left = null,
                right = null
            ),
            right = BNode(
                element = 7,
                left = null,
                right = null
            )
        )
    
        val s2 = BNode(
            element = 10,
            left = BNode(
                element = 8,
                left = null,
                right = null
            ),
            right = BNode(
                element = 12,
                left = null,
                right = null
            )
        )
    
        println(concat(s1, s2))
    }
    
    fun concat(s1: BNode, s2: BNode): BNode {
        val s1RightMostNode = rightMostNode(s1)
    
        s1RightMostNode.right = s2
    
        return s1
    }
    
    fun rightMostNode(node: BNode): BNode {
        if (node.right == null) {
            return node
        }
    
        return rightMostNode(node.right!!)
    }
    
    data class BNode(
        val element: Int,
        var left: BNode?,
        var right: BNode?,
    )
    
  22. Design a data structure that supports the following two operations:

    All operations must take O(logn)O(\log n) time on an $n$-element set.

    Solution
    fun test() {
        val ds = DataStructure()
    
        ds.insert(5)
        ds.insert(2)
        ds.insert(3)
    
        println("Median: ${ds.median()}")
        
        ds.insert(4)
        println("Median: ${ds.median()}")
    }
    
    class DataStructure {
    
        /* The head of this queue is the least element with respect to the specified ordering. */
        private val minHeap = PriorityQueue<Int>() // Represents the upper half
        private val maxHeap = PriorityQueue<Int>() // Represents the lower half
    
        fun insert(x: Int) {
            if (maxHeap.isEmpty() || x <= -maxHeap.peek()) {
                maxHeap.offer(-x)
            } else {
                minHeap.offer(x)
            }
    
            // Balance the heaps to ensure the size difference is at most 1
            if (maxHeap.size > minHeap.size + 1) {
                minHeap.offer(-maxHeap.poll())
            } else if (minHeap.size > maxHeap.size) {
                maxHeap.offer(-minHeap.poll())
            }
        }
    
        fun median(): Double {
            return if (minHeap.size == maxHeap.size) {
                //  If the number of elements is even, take the average of the middle two
                (minHeap.peek() + (-maxHeap.peek())) / 2.0
            } else {
                -maxHeap.peek().toDouble()
            }
        }
    }
    
  23. Assume we are given a standard dictionary (balanced binary search tree) defined on a set of nn strings, each of length at most ll. We seek to print out all strings beginning with a particular prefix pp. Show how to do this in O(mllogn)O(m \cdot l \cdot \log n) time, where mm is the number of strings.

    Solution

    g

  24. An array AA is called k$-unique if it does not contain a pair of duplicate elements within $k positions of each other, that is, there is no ii and jj such that A[i]=A[j]A[i] = A[j] and jik|j - i| \leq k. Design a worst-case O(nlogk)O(n \cdot \log k) algorithm to test if AA is $k$-unique.

    Solution

    g

  25. In the bin-packing problem, we are given nn objects, each weighing at most 1 kilogram. Our goal is to find the smallest number of bins that will hold the nn objects, with each bin holding 1 kilogram at most.

    Solution

    g

  26. Suppose that we are given a sequence of nn values x1,x2,...,xnx_1, x_2, ..., x_n and seek to quickly answer repeated queries of the form: given ii and jj, find the smallest value in xi,...,xjx_i, . . . , x_j. a. Design a data structure that uses O(n2)O(n^2) space and answers queries in O(1)O(1) time. b. Design a data structure that uses O(n)O(n) space and answers queries in O(logn)O(log n) time. For partial credit, your data structure can use O(nlogn)O(n log n) space and have O(logn)O(log n) query time.

    Solution
fun test() {
    val numbers1 = listOf(1, 3, 9, 2, 5)
    println(solutionA(numbers1, 2, 5))
    println(solutionA(numbers1, 1, 3))
    println(solutionB(numbers1, 2, 5))
    println(solutionB(numbers1, 1, 3))
}

fun solutionA(numbers: List<Int>, i: Int, j: Int): Int? {
    val list = Array<Array<Int?>>(numbers.size) { Array(numbers.size) { null } }

    for (p in numbers.indices) {
        var minimumSoFar = Int.MAX_VALUE

        for (q in (p+1) until numbers.size) {
            val qthNumber = numbers[q]
            if (qthNumber < minimumSoFar) {
                minimumSoFar = qthNumber
            }

            list[p][q] = minimumSoFar
        }
    }

    return list[i-1][j-1]
}

fun solutionB(numbers: List<Int>, i: Int, j: Int): Int? {
    TODO()
}
  1. Suppose you are given an input set SS of nn integers, and a black box that if given any sequence of integers and an integer kk instantly and correctly answers whether there is a subset of the input sequence whose sum is exactly kk. Show how to use the black box O(n)O(n) times to find a subset of SS that adds up to kk.

    Solution
    fun test() {
        println(findSubset(listOf(3, 5, 8, 2, 1), 6))
        println(findSubset(listOf(2, 3, 5, 8, 6, 4), 20))
    }
    
    fun findSubset(s: List<Int>, k: Int): List<Int>? {
        if (!blackBox(s, k)) return null
    
        var subset = s
        for (i in s.indices) {
            val subsetWithoutIthNumber = subset.filter { it != s[i] }
    
            if (blackBox(subsetWithoutIthNumber, k)) {
                subset = subsetWithoutIthNumber
            }
        }
    
        return subset
    }
    
    fun blackBox(integers: List<Int>, k: Int): Boolean {
        // We were not told to implement the black box, so we will hack it for the tests
        return if (k == 6) {
            integers.containsAll(listOf(1, 2, 3))
        } else return if (k == 20) {
            integers.containsAll(listOf(2, 8, 6, 4)) || integers.containsAll(listOf(3, 5, 8, 4))
        } else {
            throw IllegalArgumentException()
        }
    }
    
  2. Let A[1..n]A[1..n] be an array of real numbers. Design an algorithm to perform any sequence of the following operations:

    There are no insertions or deletions; the only change is to the values of the numbers. Each operation should take O(logn)O(log n) steps. You may use one additional array of size nn as a work space.

    Solution

    g

  3. Extend the data structure of the previous problem to support insertions and deletions. Each element now has both a ''key'' and a ''value''. An element is accessed by its key, but the addition operation is applied to the values. The ''Partial_sum'' operation is different.

    The worst-case running time should still be O(nlogn)O(n log n) for any sequence of O(n)O(n) operations.

    Solution

    g

  4. You are consulting for a hotel that has nn one-bed rooms. When a guest checks in, they ask for a room whose number is in the range [l,h][l, h]. Propose a data structure that supports the following data operations in the allotted time:

    Solution
    fun test() {
        val ds = DataStructure()
        ds.initialize(10)
        println(ds.root)
    
        println("Available in [2, 5]: ${ds.count(2, 5)}")
        val checkIn1 = ds.checkIn(2, 5)
        println("Available in [2, 5]: ${ds.count(2, 5)}")
        val checkIn2 = ds.checkIn(2, 5)
        println("Available in [2, 5]: ${ds.count(2, 5)}")
    
        ds.checkOut(checkIn2!!)
        println("Available in [2, 5]: ${ds.count(2, 5)}")
        ds.checkOut(checkIn2)
        ds.checkOut(checkIn1!!)
        println("Available in [2, 5]: ${ds.count(2, 5)}")
    }
    
    class DataStructure() {
    
        lateinit var root: BNode
    
        fun initialize(n: Int) {
            root = sortedListToBinaryTree((1..n).toList())!!
        }
    
        fun sortedListToBinaryTree(list: List<Int>): BNode? {
            if (list.isEmpty()) {
                return null
            }
    
            val mid = list.size / 2
            val root = BNode(list[mid])
            root.left = sortedListToBinaryTree(list.slice(0 until mid))
            root.right = sortedListToBinaryTree(list.slice((mid + 1) until list.size))
    
            return root
        }
    
        fun count(l: Int, h: Int): Int {
            return countInRange(l, h, root)
        }
    
        fun checkIn(l: Int, h: Int): Int? {
            val first = findFirstInRange(l, h, root)
            first?.isCheckedIn = true
            return first?.element
        }
    
        fun checkOut(x: Int) {
            find(x, root)?.isCheckedIn = false
        }
    
        fun countInRange(l: Int, h: Int, node: BNode?): Int {
            if (node == null) {
                return 0
            }
    
            val count = if (node.element in l..h && !node.isCheckedIn) 1 else 0
    
            return if (node.element in l..h) {
                count + countInRange(l, h, node.left) + countInRange(l, h, node.right)
            } else if (node.element < l) {
                count + countInRange(l, h, node.right)
            } else {
                count + countInRange(l, h, node.left)
            }
        }
    
        fun findFirstInRange(l: Int, h: Int, node: BNode?): BNode? {
            if (node == null) {
                return null
            }
    
            if (node.element in l..h && !node.isCheckedIn) {
                return node
            }
    
            return if (node.element < l) {
                findFirstInRange(l, h, node.right)
            } else if (node.element > h)  {
                findFirstInRange(l, h, node.left)
            } else {
                findFirstInRange(l, h, node.left) ?: findFirstInRange(l, h, node.right)
            }
        }
    
        fun find(x: Int, node: BNode?): BNode? {
            if (node == null) {
                return null
            }
    
            return if (node.element == x) {
                node
            } else if (node.element < x) {
                find(x, node.right)
            } else {
                find(x, node.left)
            }
        }
    }
    
    data class BNode(
        val element: Int,
        var isCheckedIn: Boolean = false,
        var left: BNode? = null,
        var right: BNode? = null,
    )
    
  5. Design a data structure that allows one to search, insert, and delete an integer XX in O(1)O(1) time (i.e., constant time, independent of the total number of integers stored). Assume that 1Xn1 \leq X \leq n and that there are m+nm + n units of space available, where mm is the maximum number of integers that can be in the table at any one time. (Hint: use two arrays A[1..n]A[1..n] and B[1..m]B[1..m].) You are not allowed to initialize either AA or BB, as that would take O(m)O(m) or O(n)O(n) operations. This means the arrays are full of random garbage to begin with, so you must be very careful.

    Solution
    fun test() {
        val ds = DataStructure(m = 5, n = 10)
        ds.insert(7)
        ds.insert(3)
        ds.insert(6)
        ds.insert(5)
        ds.insert(9)
    
        println(ds.search(6))
        println(ds.search(9))
    
        ds.delete(6)
        ds.delete(9)
    
        println(ds.search(6))
        println(ds.search(9))
    }
    
    class DataStructure(
        val m: Int,
        val n: Int
    ) {
    
        val indices = arrayOfNulls<Int>(n + 1)
        val values = arrayOfNulls<Int>(m + 1)
        var k = 0
    
        fun insert(x: Int) {
            k++
            indices[x] = k
            values[k] = x
        }
    
        fun search(x: Int): Int? {
            val index = indices[x] ?: return null
            return values[index]
        }
    
        fun delete(x: Int) {
            val xIndex = indices[x] ?: return
            if (k == 0) return
    
            val lastValue = values[k]!!
            indices[lastValue] = xIndex
            values[xIndex] = lastValue
            values[k] = null
            indices[x] = null
    
            k--
        }
    }
    
  6. What method would you use to look up a word in a dictionary?

    Solution

Use a hash table: Store the word as the key and the definition as the value.

  1. Imagine you have a closet full of shirts. What can you do to organize your shirts for easy retrieval?

    Solution

    Sort them based on specific property like color.

  2. Write a function to find the middle node of a singly linked list.

    Solution
    fun test() {
        val node1 = Node("Elvis", null)
    
        val node2 = Node("Chidera", null)
        node1.next = node2
    
        val node3 = Node("Nnajiofor", null)
        node2.next = node3
    
        val node4 = Node("Jollof", null)
        node3.next = node4
    
        val node5 = Node("List", null)
        node4.next = node5
    
        println(middleElement(node1))
    }
    
    data class Node(
        val element: String,
        var next: Node?
    )
    
    fun middleElement(head: Node): Node {
        var hare: Node? = head
        var tortoise: Node = head
    
        while (hare?.next != null) {
            hare = hare.next?.next
            tortoise = tortoise.next!!
        }
    
        return tortoise
    }
    
  3. Write a function to determine whether two binary trees are identical. Identical trees have the same key value at each position and the same structure.

    Solution
    fun test() {
        val s1 = BNode(
            element = 5,
            left = BNode(
                element = 3,
                left = null,
                right = null
            ),
            right = BNode(
                element = 7,
                left = null,
                right = null
            )
        )
    
        val s2 = BNode(
            element = 10,
            left = BNode(
                element = 8,
                left = null,
                right = null
            ),
            right = BNode(
                element = 12,
                left = null,
                right = null
            )
        )
    
        println(isIdentical(s1, s2))
        println(isIdentical(s1, s1))
        println(isIdentical(s2, s2))
    }
    
    fun isIdentical(tree1: BNode?, tree2: BNode?): Boolean {
        if (tree1 == null || tree2 == null) {
            return tree1 == tree2
        }
    
        return tree1.element == tree2.element &&
                isIdentical(tree1.left, tree2.left) &&
                isIdentical(tree1.right, tree2.right)
    }
    
    data class BNode(
        val element: Int,
        var left: BNode?,
        var right: BNode?,
    )
    
  4. Write a program to convert a binary search tree into a linked-list.

    Solution

    See Problem 14: Part of the solution involves converting a tree into a linked-list.

  5. Implement an algorithm to reverse a linked list. Now do it without recursion.

    Solution
    fun test() {
        val node1 = Node("Elvis", null)
    
        val node2 = Node("Chidera", null)
        node1.next = node2
    
        val node3 = Node("Nnajiofor", null)
        node2.next = node3
    
        println(reverse(node1))
    }
    
    fun reverse(head: Node): Node {
        var currentNode: Node? = head
        var previousNode: Node? = null
    
        while (currentNode != null) {
            val nextNode = currentNode.next
            currentNode.next = previousNode
            previousNode = currentNode
            currentNode = nextNode
        }
    
        return previousNode!!
    }
    
    data class Node(
        val element: String,
        var next: Node?
    )
    
  6. What is the best data structure for maintaining URLs that have been visited by a web crawler? Give an algorithm to test whether a given URL has already been visited, optimizing both space and time.

    Solution

    A dictionary that acts as a set:

    • When a URL is visited, it's added to the dictionary.
    • To check if a URL has been visited, check if the dictionary contains the URL as a key.
  7. You are given a search string and a magazine. You seek to generate all the characters in the search string by cutting them out from the magazine. Give an algorithm to efficiently determine whether the magazine contains all the letters in the search string.

    Solution
    fun test() {
        val node1 = Node("Elvis", null)
    
        println(containsAllCharacters("Elvis", "The Elfs are very happy"))
        println(containsAllCharacters("Elvis", "The Elfs are very happy because it is christmas"))
    }
    
    fun containsAllCharacters(string: String, magazine: String): Boolean {
        val map = mutableMapOf<Char, Int>()
    
        string.forEach { char ->
            map[char] = map.getOrDefault(char, 0) + 1
        }
    
        var matched = 0
        magazine.forEach { char ->
            val count = map[char] ?: 0
    
            if (count > 0) {
                matched++
                map[char] = count - 1
            }
    
            if (matched == string.length) {
                return true
            }
        }
    
        return false
    }
    
  8. Reverse the words in a sentence—that is, “My name is Chris” becomes “Chris is name My.” Optimize for time and space.

    Solution
    fun test() {
        println(reverse("My name is Chris"))
    }
    
    fun reverse(sentence: String): String {
        val words = sentence.split(" ")
        var reversedSentence = ""
    
        var i = words.lastIndex
        while (i >= 0) {
            val word = words[i]
            reversedSentence += word
            i--
    
            if (i >= 0) {
                reversedSentence += " "
            }
        }
    
        return reversedSentence
    }
    
  9. Determine whether a linked list contains a loop as quickly as possible without using any extra storage. Also, identify the location of the loop.

    Solution
    fun test() {
        val node1 = Node("Elvis", null)
    
        val node2 = Node("Chidera", null)
        node1.next = node2
    
        val node3 = Node("Nnajiofor", null)
        node2.next = node3
    
        val node4 = Node("Jollof", null)
        node3.next = node4
    
        val node5 = Node("List", null)
        node4.next = node5
    
        node5.next = node3
    
        println(detectLoopLocation(node1)?.element)
    }
    
    data class Node(
        val element: String,
        var next: Node?
    )
    
    /**
    * https://en.wikipedia.org/wiki/Cycle_detection#:~:text=Floyd's%20cycle%2Dfinding%20algorithm%20is,The%20Tortoise%20and%20the%20Hare.
    */
    fun detectLoopLocation(head: Node): Node? {
        var hare: Node? = head
        var tortoise: Node? = head
    
        while (hare != null) {
            hare = hare.next?.next
            tortoise = tortoise?.next
    
            if (hare == tortoise) {
                break
            }
        }
    
        if (head.next != null && hare == tortoise) {
            hare = head
            while (hare != tortoise) {
                hare = hare?.next
                tortoise = tortoise?.next
            }
    
            return tortoise!!
        }
    
        return null
    }
    
  10. You have an unordered array XX of nn integers. Find the array MM containing nn elements where MiM_i is the product of all integers in XX except for XiX_i. You may not use division. You can use extra memory. (Hint: there are solutions faster than O(n2)O(n^2).)

    Solution
    fun test() {
        println(transform(arrayOf(3, 5, 4)).toList())
        println(transform(arrayOf(2, 3, 4, 5, 6)).toList())
    }
    
    fun transform(x: Array<Int>): Array<Int> {
        val prefixProducts = Array(x.size) { 0 }
        val suffixProducts = Array(x.size) { 0 }
    
        var prefixProduct = 1
        x.forEachIndexed { i, xi ->
            prefixProducts[i] = prefixProduct
            prefixProduct *= xi
        }
    
        var suffixProduct = 1
        var i = x.lastIndex
        while (i >= 0) {
            val xi = x[i]
            suffixProducts[i] = suffixProduct
            suffixProduct *= xi
            i--
        }
    
        return Array(x.size) {
            prefixProducts[it] * suffixProducts[it]
        }
    }
    
  11. Give an algorithm for finding an ordered word pair (e.g. “New York”) occurring with the greatest frequency in a given webpage. Which data structures would you use? Optimize both time and space.

    Solution
    fun test() {
        println(findOrderedWordPairWithMaxFrequency("New york is a great city. I love new york."))
        println(findOrderedWordPairWithMaxFrequency("My name is Elvis Chidera. Elvis Chidera is me."))
    }
    
    /*
    Punctuation marks can mess up the algorithm. Only "." is handled.
     */
    fun findOrderedWordPairWithMaxFrequency(text: String): Pair<String, String> {
        val words = text.replace(".", "").split(" ")
        if (words.size <= 1) throw IllegalArgumentException("Text is empty or only has one word")
    
        val wordPairs = mutableMapOf<Pair<String, String>, Int>()
    
        var i = 1
        while (i < words.size) {
            val previousWord = words[i - 1]
            val currentWord = words[i]
            val wordPair = previousWord to currentWord
    
            wordPairs[wordPair] = wordPairs.getOrDefault(wordPair, 1) + 1
            i++
        }
    
        var maxFrequency = 0
        var wordPairWithMaxFrequency: Pair<String, String>? = null
        wordPairs.forEach { (wordPair, frequency) ->
            if (frequency > maxFrequency) {
                maxFrequency = frequency
                wordPairWithMaxFrequency = wordPair
            }
        }
    
        return wordPairWithMaxFrequency!!
    }
    

Chaper 4: Sorting

This chapter discusses sorting, algorithm design paradigms espoused by sorting algorithms and how sorting can be applied in solving other problems.

Applications of sorting

The first algorithm design paradigm is the application of sorting as an initial step to simplify another computational problem. e.g.

Sorting pragmatics

Pairwise comparison (i.e. is aa >> or << or == to $b$) is typical abstracted away into a comparison function that is then passed to the sorting algorithm. This allows the sorting algorithm to be independent of application-specific comparison concerns like:

An example comparison function:

/**
* Returns a negative integer if `e1` is less than `e2`.
* Returns zero if `e1` is equal `e2`.
* Returns a positive integer if `e1` is greater than `e2`.
*/
fun compare(e1: Element, e2: Element): Int {

}

Stable sorting algorithms maintain the relative order of elements with equal keys. That is, a sorting algorithm is stable if whenever there are two elements RR and SS with the same key and with RR appearing before SS in the original list, RR will appear before SS in the sorted list.

Few fast algorithms are naturally stable. Stability can be achieved for any sorting algorithm by adding the initial position as a secondary key.

When equal elements are indistinguishable, such as with integers, or more generally, any data where the entire element is the key, stability is not an issue.

Heap: Data structure

A heap is a tree-based data structure that satisfies the heap property:

The heap is one maximally efficient implementation of a priority queue ADT. In a heap, the highest (or lowest) priority element is always stored at the root. A heap is a useful data structure when it is necessary to repeatedly remove the object with the highest (or lowest) priority.

However, a heap is not a sorted structure; it can be regarded as being partially ordered: there is no implied ordering between siblings or cousins and no implied sequence for an in-order traversal. The heap relation mentioned above applies only between nodes and their parents, grandparents, etc.

Implementation

Heaps are usually implemented with an array:

For a binary heap, in the array:

This can be generalized into:

Given a node at index ii,

Its left child is at index 2i+12i + 1

Its right child is at index 2i+22i + 2

And its parent is at index (i1)/2\lfloor (i - 1) / 2 \rfloor

This simple indexing scheme makes it efficient to move "up" or "down" the tree.

This implicit structure trades space efficiency for flexibility. e.g. With pointers, we can move subtrees around by changing a single pointer.

To avoid potentially requiring a 2n12^n - 1 array to store an n$-element heap, it's required that the binary heap is complete with its element inserted from left to right. This structural constraint allows us to store the $n$-element heap with the first $n indices of the array. When a heap is a complete binary tree, it has the smallest possible height — a heap with NN nodes and aa branches for each node always has logaN\log_a N height.

Heap as an array

/**
 * Put elements of [array] in heap order, in-place.
 *
 * Consider the array in reverse order, starting from the last (nth) position. It represents a leaf of the tree and
 * so trivial obeys the heap property (i.e. it's larger than its non-existent children). The same is the case for
 * the last n/2 positions in the array, because all are leaves.
 *
 * If we continue to walk backwards through the array we will eventually encounter an internal node with children.
 * This element may not obey the heap property (because it might be smaller than its children),
 * but its children represent well-formed (if small) heaps. This situation is exactly what [siftDown] was designed
 * to do: given a "damaged" binary heap, where the max-heap property (no child is greater than its parent) holds
 * everywhere except possibly between the root node and its children, repair it to produce an undamaged heap.
 *
 * To see that this takes O(n) time, count the worst-case number of [siftDown] iterations.
 * The last half of the array requires zero iterations, the preceding quarter requires at most one iteration,
 * the eighth before that requires at most two iterations, the sixteenth before that requires at most three, and so on.
 */
private fun heapify(array: ArrayList<Int>) {
    // Start is initialized to the first leaf node
    // Find the parent of the last element.
    var start = iParent(array.lastIndex) + 1

    while (start > 0) {
        // Go to the last non-heap node
        start -= 1
        // Sift down the node at index 'start' to the proper place such that all nodes below the start index are in heap order
        siftDown(array, start, array.lastIndex)
    }

    // After sifting down the root all nodes/elements are in heap order.
}

private fun siftUp(array: ArrayList<Int>, index: Int) {
    var index = index

    while (index > 0) {
        val parentIndex = iParent(index)

        if (array[parentIndex] < array[index]) {
            swap(array, parentIndex, index)
            index = parentIndex
        } else {
            return
        }
    }
}

/**
 * Repair the heap whose root element is at index 'start', assuming the heaps rooted at its children are valid.
 *
 * The number of iterations in any one [siftDown] call is bounded by the height of the tree, which is ⌊log2 n⌋ = O(log n).
 */
fun siftDown(array: ArrayList<Int>, rootIndex: Int, endIndex: Int) {
    var rootIndex = rootIndex
    while (iLeftChild(rootIndex) <= endIndex) { // While the root has at least one child
        var biggerChildIndex = iLeftChild(rootIndex)

        // If there is right child and that child is greater
        if (iRightChild(rootIndex) <= endIndex && array[biggerChildIndex] < array[iRightChild(rootIndex)]) {
            biggerChildIndex = iRightChild(rootIndex)
        }

        if (array[rootIndex] < array[biggerChildIndex]) {
            swap(array, rootIndex, biggerChildIndex)
            rootIndex = biggerChildIndex
            // Repeat to continue sifting down the child now.
        } else {
            // The root holds the largest element. Since we may assume the heaps rooted at the children are valid,
            // this means that we are done.
            return
        }
    }
}

private fun swap(array: ArrayList<Int>, i: Int, j: Int) {
    val temp = array[i]
    array[i] = array[j]
    array[j] = temp
}

fun iParent(i: Int): Int {
    return floor(i / 2.0).toInt()
}

fun iLeftChild(i: Int): Int {
    return (i * 2) + 1
}

fun iRightChild(i: Int): Int {
    return (i * 2) + 2
}

data class Heap(private val array: ArrayList<Int>) {

    init {
        heapify(array)
    }

    /**
     * Add the new element at the end of the heap, in the first available free space.
     * If this violates the heap property, sift up the new element until the heap property has been reestablished.
     * Since the heap is balanced, the height is $log_2 n$ and hence each insertion costs at most $O(log_2 n)$.
     */
    fun insert(element: Int) {
        array.add(element)
        siftUp(array, array.lastIndex)
    }

    /**
     * Remove the root and insert the last element of the heap in the root.
     * If this violates the heap property, sift down the new root to reestablish the heap property.
     */
    fun pop(): Int {
        val root = array[0]
        val lastChild = array.removeLast()
        array[0] = lastChild
        siftDown(array, 0, array.lastIndex)
        return root
    }

    /**
     * Remove the root and put the new element in the root and sift down.
     * When compared to pop followed by insert, this avoids a sift up step.
     */
    fun replaceRoot(element: Int) {
        array[0] = element
        siftDown(array, 0, array.lastIndex)
    }
}

Heapsort: Fast sorting via data structures

Heapsort is an in-place comparison-based sorting algorithm which can be thought of as "an implementation of selection sort using the right data structure." Like selection sort, heapsort divides its input into a sorted and an unsorted region, and it iteratively shrinks the unsorted region by extracting the largest element from it and inserting it into the sorted region. Unlike selection sort, heapsort does not waste time with a linear-time scan of the unsorted region; rather, heap sort maintains the unsorted region in a heap data structure to efficiently find the largest element in each step.

Although somewhat slower in practice on most machines than a well-implemented quicksort, it has the advantages of very simple implementation and a more favorable worst-case O(nlogn)O(n \log n) runtime. Most real-world quicksort variants include an implementation of heapsort as a fallback should they detect that quicksort is becoming degenerate.

Heapsort is an example of the second algorithm design paradigm: Use an appropriate data structure.

/**
 * The steps are:
 *
 * 1. Call the [heapify] function on the array. This builds a heap from an array in O(n) operations.
 * 2. Swap the first element of the array (the largest element in the heap) with the final element of the heap. Decrease the considered range of the heap by one.
 * 3. Call the [siftDown] function on the array to move the new first element to its correct place in the heap.
 * 4. Go back to step (2) until the remaining array is a single element.
 *
 * The [heapify] operation is run once, and is O(n) in performance.
 * The [siftDown] function is called n times and requires O(log n) work each time, due to its traversal starting from the root node.
 * Therefore, the performance of this algorithm is O(n + n log n) = O(n log n).
 */
fun heapsort(array: ArrayList<Int>): ArrayList<Int> {
    // Build the heap in the array so that largest value is at the root
    heapify(array)
    var endIndex = array.lastIndex

    // The following loop maintains the invariants that every element between 0 to endIndex is a heap, and
    // every element beyond endIndex is greater than everything before it (in sorted order).
    while (endIndex > 0) {
        // [0] is the root and largest value. The swap moves it in front of the sorted elements.
        swap(array, 0, endIndex)
        // The heap size is reduced by one.
        endIndex--
        // The swap ruined the heap property, so restore it.
        siftDown(array, 0, endIndex)
    }

    return array
}