- describe the differences between Linear and Binary Search algorithms
- define what runtime complexity is, differentiate between various classifications and categorize the performance of an algorithm using Big O notation
- distinguish when to use, classify the performance, and implement code to conduct classic iterative sorting algorithms
In this project, you will be implementing many different solutions to the same problem: sort a list of integers in ascending order. You will also be using your newfound knowledge of complexity analysis to evaluate each implementation for efficiency.
Although sorting may seem like a trivial task, this project allows you to begin practicing classifying and categorizing the performance of different algorithms. Practicing your ability to classify the performance of these simple algorithms will make it easier for you to classify more sophisticated algorithms as you continue your journey of mastering Computer Science fundamentals. It will also help you to see that even trivial computation tasks embedded in your programs are essential for you to understand. Understanding these details is part of what will make you an excellent programmer and developer.
As you begin interviewing for jobs, you will often be expected to be able to implement, classify, and deeply understand these classic sorting algorithms.
- Open up the iterative_sorting directory
- Read through the descriptions of the
bubble_sort
andselection_sort
algorithms - Implement
bubble_sort
andselection_sort
in iterative_sorting.py - Test your implementation by running
test_iterative.py
- Implement all the methods in the searching.py file in the searching directory.
- Test your implementation of the searching methods by running
test_searching.py
.
- Implement the
count_sort
algorithm in the iterative_sorting directory.
Make sure to test your implementations by running test_iterative.py
and test_searching.py
. To test the stretch goal, you need to uncomment the tests for count_sort
in test_iterative.py
.
Generally speaking, at work and in interviews, people are more interested in time complexity. That said, space complexity, or how much additional space is required to process n elements of data, is also very important.
Space complexity does not include the space required to hold the data you are going to process. It only includes the additional space requirements of your algorithm.
Here's an example with O(1)
space requirements:
def alg(data):
result = 0
for v in data:
result += v
return result
Clearly the time complexity is O(n)
. As list data
grows, the time it takes
to complete the algorithm grows proportionately.
But the space complexity is O(1)
. The additional space required to complete
the algorithm was:
result
:O(1)
v
:O(1)
And neither of those change in size regardless of how big list data
is. data
could have a zillion elements, and the algorithm would still only require space
for result
and v
.
What about this:
def alg(data):
new_data = data.copy() # <-- Added this line
result = 0
for v in new_data:
result += v
return result
Here we have more space allocated.
result
:O(1)
v
:O(1)
new_data
:O(n)
new_data
gets bigger as data
gets bigger, so it's O(n)
. So we have:
O(1) + O(1) + O(n)
space requirements. O(n)
dominates, so the final result
is O(n)
.
Where things get really tricky is when you have a recursive call. Each call to the function allocates space for all the local variables.
And it gets doubly tricky if you're making a copy of the data each call, like with a slice.
Here's the same algorithm as before that adds all the elements in a list, except
this one does it recursively. Recall that the initial iterative version was
O(1)
space complexity.
def alg(data):
if data == []:
return 0
first_val = data[0]
return first_val + alg(data[1:])
Every time we make a call into alg()
, we get new space for local variables. So
any local variable that was O(1)
on its own, now gets n copies and becomes
O(1*n)
or O(n)
.
But that's not all. When we slice data
on the recursive call, we get a new
list. So that means every call we're getting another copy of the list, and each
copy is O(n)
space as well. And we're making n recursive calls, so we need
n copies of O(n)
space, which comes to O(n*n)
or O(n^2)
space.
Sure, it's not a complete copy of the list. It's missing the first element. So each call has a shorter and shorter list, like:
[ 1, 2, 3, 4 ] [ 2, 3, 4 ] [ 3, 4 ] [ 4 ] [ ]So that means the first call wasn't really
O(n)
, but was more likeO(0.8*n)
, and the second wasO(0.6*n)
, and so on.But recall that we drop constants with Big-O, so those still just become
O(n)
.
So we have n recursive calls.
first_val
on its own is O(1)
. But we recurse n times, so it becomes
O(1*n)
or just O(n)
.
Each slice of data
on its own is O(n)
. But we recurse n times, so it
becomes O(n*n)
or O(n^2)
.
So the total space complexity for this algorithm is:
O(n) + O(n^2)
. The O(n^2)
dominates, so the final space complexity is just
O(n^2)
.
Again, compare to the O(1)
space complexity of the initial iterative solution.
In some languages, notably
Lisp and other
functional programming
languages, you can write
recursive solutions with O(1)
space complexity. These languages take advantage
of a feature known as tail call
optimization to make this possible.
By coding things correctly, the language can automatically convert your
recursive solution into an iterative solution. Stock C and Python do not support
tail call optimization.
Yes.
Generally the built-in functionality is better tested and better optimized than something you could write.
That said, there might be times you want to write your own versions. Maybe the built-in doesn't do exactly what you need. Or maybe you want to rewrite the built-in as a learning exercise, for example.
Memory: in-place typically takes less memory since you're reusing the original storage for the list to hold the finally sorted list. If you allocate more lists to hold the result, that takes more memory.
Code clarity: in-place sometimes has more complex code making the algorithm harder to understand.
As an example, here's a non-in-place Quicksort, which seems to be generally clearer than the in-place variant, but uses far more memory:
def partition(data):
left = []
pivot = data[0]
right = []
for v in data[1:]:
if v <= pivot:
left.append(v)
else:
right.append(v)
return left, pivot, right
def quicksort(data):
if data == []:
return data
left, pivot, right = partition(data)
return quicksort(left) + [pivot] + quicksort(right)