Sorting Algorithms

Projected Time

5-6 hours

Prerequisites

Motivation

Sorting is important in programming for the same reason it is important in everyday life. It is easier and faster to locate items in a sorted list than unsorted. Sorting algorithms can be used in a program to sort an array for later searching or writing out to an ordered file or report.(by Siddharth Khuntwal)

For example bubble sort, quick sort, and selection sort all take the same input and produce the same output but at different speeds. This is a nice way to show how different approaches can yield more efficient algorithms using approaches such as divide and conquer. On top of that, sorting algorithms don’t require advanced data structures knowledge to understand. If you can understand arrays you can understand sorting algorithm examples.(by cma)

Objectives

Participants will be able to:

Materials and Useful Videos

Pre-Lesson Warm Up and Game (30 minutes)

Lesson (3 hours)

Remember how we can sort cards lots of different ways? Let’s talk about all the ways computers sort Lists.

Let’s start with a Bad Way To Sort:

BogoSort (10 minutes)

Say we want to sort a deck of cards:

  1. Toss them in the air!
  2. Pick them up one by one.
  3. Are they sorted?
  4. If not, repeat lines 1 - 3

Let’s talk about the Complexity of BogoSort. Here’s the pseudocode:

while not Sorted(a) 
   Shuffle(a)

The best case?

O(n) where n is the length of the list, because we have to check that the list is sorted.

The worst case?

O(????) because this could NEVER END if we get really unlucky.

Hopefully you can see why we never want to use this algorithm again.

Bubble Sort (30 minutes)

Thanks, Obama

How does it work?

  1. Compare consecutive pairs of elements.
  2. Swap elements in the pair so that the smaller one is first.
  3. When we reach the end of the list, start over!
  4. Stop when no more swaps have been made.

Demonstration video: Watch this Video (useful in slo-mo)

Let’s watch a folk dance interpretation of Bubble Sort: Watch this video, It is also available in the materials section

Let’s break into groups again and try out the sorting algorithm on our own decks of cards!

What’s the Complexity of Bubble Sort?

function bubble_sort(a)
{
    let swap;
    let n = a.length-1;
    let x=a;
    do {
        swap = false;
        for (let i=0; i < n; i++)
        {
            if (x[i] < x[i+1])
            {
               let temp = x[i];
               x[i] = x[i+1];
               x[i+1] = temp;
               swap = true;
            }
        }
        n--;
    } while (swap);
    return x; 
}   

We have nested loops: the while-loop and the for-loop! The outer loop passes until there are no more swaps. Thus the runtime is O(n^2).

Group question: what state would the list be in for Bubble Sort to have the worst possible performance?

Selection Sort (15 minutes)

How does it work?

  1. First step:
    1. Extract the smallest element.
    2. Swap it with the element at index 0.
  2. Next step:
    1. In the remaining unsorted sublist extract the smallest element.
    2. Swap it with the element at index 1.
  3. At the i’th step, the first i elements of the list are guaranteed to be sorted.

Demonstration video: Watch this video

What’s the Complexity of Selection Sort?

function selection_sort(a){
  for( let i = 0; i < a.length; i++ ){
    let small = i;
    for( let j = i + 1; j < a.length; j++ ){
      if( a[j] < a[small]){
        small = j;
      }
    }
    if(i !== small){
      let temp = a[i];
      a[i] = a[small];
      a[small] = temp;
    }
  }
  return a;
}

Again, there’s a nested loop, so worst-case runtime is O(n^2)!

Can we do better than O(n^2)?????

Merge Sort (2 hours)

Merge Sort is a divide and conquer algorithm. A divide and conquer algorithm works by recursively breaking down a problem into two or more sub-problems of the same or related type, until these become simple enough to be solved directly. The solutions to the sub-problems are then combined to give a solution to the original problem.

What are we dividing and what are we conquering here?

Well, we divide the list into lots of sublists. In fact, we divide them into lists that are of size 1! At that point, we know a list of size 1 is sorted! This isn’t intuitive, right? Let’s dig in…

  1. If a list is of length 0 or 1, it’s already sorted!
  2. If a list has more than one element, split it into two lists, and sort each.
  3. Merge sorted sublists.
    1. Look at the first element of each, move smaller to the end of the result.
    2. When one list is empty, just copy the rest of the other list.

Let’s review this video: Watch this video, it is also available in the materials section

This will be tough, but let’s try implementing it ourselves!

Let’s try just the merge (a, left, right, middle) function first.

function merge (a, left, right, middle){
    n1 = middle - left +1;
    n2 = right - middle;
    let  L = [];
    let R = [];
    for(let i =0;i<n1;i++){
        L[i] = a[left + i ];
    }
    for(let j=0;j<n2;j++){
        R[j] = a[middle+j+1];
    }
    i = 0;    
    j = 0 ;   
    let k = left;
    while (i < n1 && j < n2)
    {
        if (L[i] <= R[j])
            a[k] = L[i++];
        else
            a[k] = R[j++];
        k++;   
    }
    while (i < n1)
    {
        a[k] = L[i];
        i++;
        k++;
    }
    while (j < n2)
    {
        a[k] = R[j];
        j++;
        k++;
    }
}

Can we merge this with sample input? What if left = [1,3] and right = [2, 4]?

After an hour, let’s complete the merge_sort function!

function merge_sort (a, left, right){
    if(left < right){
        let middle = Math.floor((left +right)/2);
        merge_sort(a,left,middle);
        merge_sort(a,middle+1,right);
        merge(a,left,right,middle);
    }
}

Let’s test it with this list: testList = [1,3,5,7,2,6,25,18,13]

Independent Practice (2 hours)

Spend 3 hours writing steps for yourself on how to go about writing each algorithm: Bubble, Selection, Merge. For example:

Name of Algorithm

  1. Create this function and a helper function.
  2. Flesh out the first function by creating variables and then using them in these ways.
  3. Call helper function.
  4. Flesh out the helper function by creating these variables and then using them in these ways.
  5. Return helper result to first function.
  6. This is why it worked!

If you’re curious to read more there are lots of good resources out there!

MIT Intro to CS Slides on Sorting

Harvey Mudd Lesson Plan on Sorting

SparkNotes on Sorting

https://www.toptal.com/developers/sorting-algorithms/

http://sorting.at/

Challenge

  1. Find out what Radix Sort is.
  2. Find one more sorting algorithm we didn’t mention.

Check for Understanding

Supplemental Resources

Attribution

Thanks to the MIT OpenCourseWare site for Sorting Algorithm Code Examples! https://ocw.mit.edu/courses/electrical-engineering-and-computer-science/6-0001-introduction-to-computer-science-and-programming-in-python-fall-2016/lecture-slides-code/lec12_sorting.py