Binary Searching

Binary search is an improvement over linear searching that works only if the data in the array are sorted beforehand.

Suppose we have the following array data shown under the array indices:

10  20  30  40  50  60  70  80  90  100 115 125 135 145 155 178 198

Binary search works by keeping track of the midpoint (mid) and the minimum (min) and maximum (max) index positions where the item might be.

If we are looking for a number, say, 115, here is a visual on how we might go about it. We display the indices over the data being considered. Here min and max are the smallest and largest index to still consider. A textual explanation follows the visual:

   0   1   2   3   4   5   6   7   8   9  10  11  12  13  14  15  16
  10  20  30  40  50  60  70  80  90 100 115 125 135 145 155 178 198
min=0 max=16 mid=8
                                     100 115 125 135 145 155 178 198
min=9 max=16 mid=12
                                     100 115 125
min=9 max=11 mid=10
Item 115 found at position 10
  • We start by testing the data at position 8. 115 is greater than the value at position 8 (100), so we assume that the value must be somewhere between positions 9 and 16.

  • In the second pass, we test the data at position 12 (the midpoint between 9 and 16). 115 is less than the value at position 12, so we assume that the value must be somewhere between positions 9 and 11.

  • In the last pass, we test the value at position 10. The value 115 is at this position, so we’re done.

So binary search (as its name might suggest) works by dividing the interval to be searched during each pass in half. If you think about how it’s working here with 17 items. Because there is integer division here, the interval will not always be precisely half. it is the floor of dividing by 2 (integer division, that is).

With the data above, you see that the algorithm determined the item within 3 steps. To reduce to one element to consider, it could be 4 or 5 steps. Note that \(4 < log_2 17 < 5\).

Now that we’ve seen how the method works, here is the code in binary_searching/binary_searching.cs that does the work:

 1/// Return the index of item in a non-empty sorted array data,
 2/// or return -1 if item is not in the array.
 3public static int IntArrayBinarySearch(int[] data, int item)
 4{
 5   int min = 0, max = data.Length-1;
 6   while(min <= max) {
 7      int mid = (min+max) / 2;
 8      if (data[mid] == item)
 9         return mid;
10      if (item > data[mid])
11         min = mid + 1;
12      else
13         max = mid - 1;
14   }
15   return -1;
16}

Here’s a quick explanation, because it largely follows from the above explanation.

  • Line 5: Initially item could be anywhere in the array, so minimum is at position 0 and maximum is at position N-1 (data.Length - 1).

  • The loop to make repeated passes over the array begins on line 6. We can only continue searching if there is some data left to consider (min <= max).

  • Line 7 does just what we expect: It calculates the median position (mid).

  • It is always possible that we’ve found the item, which is what we test on line 8, and return with our answer if we found it.

  • Lines 10-13: If not, we continue. If the item is greater than the value at this mid position, we know it is in the “upper half”. Otherwise, it’s in the “lower half”.

  • Line 15: Otherwise the binary search loop terminates, and we return -1 (to indicate not found). The -1 value is a commonly-returned indicator of failure in search operations (especially on arrays, lists, and strings), so we use this mostly out of respect for tradition. It makes particular sense, because -1 is not within the index set of the array (which starts at 0 in C# and ends at data.Length - 1.

Of course we generally would be searching in an array with multiple elements. It is still important to check edge cases: Check that the code correctly returns -1 if the array has length 0 (a legal length).

Similar to linear searching, we provide a main program that tests it out. in binary_searching/binary_searching_demo.cs. It uses an elaboration of binary search that prints out the steps visually, as in the introduction to this section. It also references previous example projects: functions from files searching/extract_from_string.cs and sorting/sorting.cs.

 1   string input = UI.PromptLine(
 2      "Please enter some comma/space separated integers: ");
 3   int[] data = ExtractFromString.IntsFromString(input);
 4   Sorting.IntArrayShellSortBetter(data);
 5   string prompt =
 6      "Please enter a number to find (empty line to end): ";
 7   input = UI.PromptLine(prompt);
 8   while (input.Length != 0) {
 9      int searchItem = int.Parse(input);
10      int foundPos = BinarySearching.IntArrayBinarySearchPrinted(
11                        data, searchItem);
12      if (foundPos < 0)
13         Console.WriteLine("Item {0} not found", searchItem);
14      else
15         Console.WriteLine("Item {0} found at position {1}",
16                           searchItem, foundPos);
17      input = UI.PromptLine(prompt);
18   }

Elaborated Binary Search Exercise

Even if you do not find item in a binary search, it is sometimes useful to know where item lies in relation to the array elements. It could be before the first element, in between two elements, or after the last element. Suppose N is the (positive) array length. Instead of just returning -1 if item is not in the array, return

  • -1 if item < data[0]

  • -(k+1) if data[k-1] < item < data[k]

  • -(N+1) if data[N-1] < item

Modify binary_searching/binary_searching.cs into binary_searching2.cs so this extra information is returned (and indicated clearly in a main testing program derived from binary_searching/binary_searching_demo.cs). This should not require a change to the while loop, nor require any added loop.