In a merge sort, the basic idea is to:
Input M E R G E S O R T E X A M P L E Sort Left Half E E G M O R R S Sort Right Half A E E L M P T X Merge Results A E E E E G L M M O P R R S T X
Being "smaller" lists, the sorting of the left and right halves are easily done recursively, until they become lists of only one element -- which are trivially already sorted.
The below diagram shows the recursive breakdown of a list of integers (up to the center row) and merging together of various pairs of smaller lists created into the final sorted array. The green arrows highlight the work done just to sort the first half of the original list (shown in pink).
But how do we "merge" two ordered subarrays into a longer ordered array?
Imagine a teacher has two stacks of papers, each already alphabetized by the student names that appear at the top of the papers, and now they need to be merged together into a single alphabetized stack. The first stack has a paper by Bob on the top, while the second has a paper by Don on top. Which paper needs to be the first in the merged stack? "B" comes before "D", so we take Bob's paper off the first stack and place it face down on the table, as the first paper of our merged stack. This reveals Chad's paper, now sitting on the top of the first stack. Which paper should be next in your merged stack -- Chad's or Don's? Well "C" comes before "D", so you take Chad's paper off of the first stack, flip it over, and place it on top of Bob's. This reveals a paper by Emily, now at the top of the first stack. Which do you choose to put face down on the table now -- Emily's or Don's? Well, "D" comes before "E", so this time, take Don's paper off the second stack, flip it over ond put it on top of Chad's. This reveals Frederica's paper, etc. That's really about all there is to it -- you just keep picking papers to add to the merged pile in this way until one of your two stacks runs out, at which point you can flip over the entire remaining stack, and add it to your merged stack.
The important thing about the process described above is that we need some table space to put the two sorted half-stacks as we merge them into a single sorted stack. Similarly, when we implement the same algorithm in code to merge two sub-arrays, we'll want to have access to a second auxiliary array large enough to hold these two sub-arrays. If we create an array as long as the one we are attempting to sort at the very beginning of the entire process (i.e., before any recursive calls have happened), we will only have to create this extra space once. For each merge needed, we just copy all of the elements from the two sub-arrays we are merging to the auxiliary array as a temporary place to hold them, and then merge them back into the same positions in the original array, but in the proper order this time.
To see how a single merge is accomplished in code, let us assume the original array to be sorted is held in an instance variable a[]
and we have also defined an auxiliary array instance variable aux[]
. We further suppose we have the standard "helper" instance method less(v,w)
for comparing the Comparable
items v
and w
.
It will be nice if the merge()
method we are to write can be invoked on whatever parts of the original array we need over the course of the entire sorting algorithm. As such, let us assume we are trying to merge two sections of the array that span from index lo
to index hi
, where one extends from index lo
to index mid
, and the other extends from index mid+1
to index hi
, where mid=(lo+hi)/2
. Consequently, we pass lo
, mid
, and hi
to the method as parameters. Of course, we also assume these two sections have already been sorted by earlier (recursive) calls to this same method..
First, we copy these elements from the original array to the auxiliary array, and then we do the following:
i
and j
, respectively. Initially, these are then i = lo
and j = mid+1
aux[i]
or aux[j]
) to a[k]
where k=lo
, initially.i
or j
) associated with the half from which we copied, and also increment k
Here's an implementation in Java:
private void merge(int lo, int mid, int hi) { // make a copy in the auxiliary array... for (int k = lo; k <= hi; k++) { aux[k] = a[k]; } // merge back to original array... int i = lo; int j = mid + 1; for (int k = lo; k <= hi; k++) { if (i > mid) a[k] = aux[j++]; // left half exhausted, copy from right else if (j > hi) a[k] = aux[i++]; // right half exhausted, copy from left else if (less(aux[j],aux[i])) a[k] = aux[j++]; else a[k] = aux[i++]; } }
Once we have the merge method written, implementing the rest of the algorithm on the elements from lo
to hi
is easy. We check to see if we are in the base case for the recursion first (i.e., a subarray of one element, where hi
$\le$ lo
). If we are, we're done. If not, we need to calculate the value of mid
, make our two recursive calls to sort the left and right halves, and then merge these two halves together:
private void mergeSort(int lo, int hi) { if (hi <= lo) return; // are we in the base case? int mid = lo + (hi - lo) / 2; mergeSort(lo, mid); // sort left half mergeSort(mid+1, hi); // sort right half merge(lo, mid, hi); // merge the two sorted halves together }
The merge sort is a beautiful and incredibly fast, divide-and-conquer recursive algorithm for sorting. That said, there are still things we can tweak to make it run even faster:
Use insertion sort for small subarrays. Merge sort shines when applied to large lists, but simply has too much in the way of overhead costs relative to the actual sorting it accomplishes when applied to tiny subarrays. On the other hand, insertion sorts are able to sort tiny subarrays quickly, but slow down to a crawl for large lists. Why not have the best of both worlds? Before calculating mid
and making the recursive calls to your sorting method, first check to see if hi
is less than low
plus some cutoff value (typically $\approx 10$). If it is -- use insertion sort for this span in the array and move on with life. If not, let the merge sort do its thing.
sort()
, only do the merge sort if the element at index mid+1
is less than the element at index mid
. Remember, the two sub-arrays are themselves already ordered -- so if these two elements are also in order, everything is.
Eliminate the copy to the auxiliary array. Imagine strumming the strings of a guitar always in the top-to-bottom direction versus strumming the strings first from top-to-bottom, then back from the bottom to the top, then top-to-bottom again, etc. Reversing the direction each time should double the speed of your strumming, as all that time getting your hand back to the top string for another strum is not wasted.
The same principle can be applied to the merge sort. Switch the role of the initial and auxiliary array in each recursive call, and avoid all that extra copying used only to always merge in the same direction (i.e., back into the initial array)! Assuming we initialize aux[]
to a[]
before the recursive calls, here's how to do this quickly in code:
private void merge(Item[] a, Item[] aux, int lo, int mid, int hi) { int i = lo; int j = mid+1; for (int k = lo; k <= hi; k++) { if (i > mid) aux[k] = a[j++]; else if (j > hi) aux[k] = a[i++]; else if (less(a[j], a[i])) aux[k] = a[j++]; // <-- notice we always merge from else aux[k] = a[i++]; // a[] to aux[] here; } } private void sort(Item[] a, Item[] aux, int lo, int hi) { if (hi <= lo) return; int mid = lo + (hi - lo) / 2; sort (aux, a, lo, mid); //<-- note in the two calls to sort(), aux[] comes sort (aux, a, mid+1, hi); // before a[], but in sort's method signature and merge(a, aux, lo, mid, hi); } // in merge(), they are reversed. this is how // aux[] and a[] get repeatedly switched.