How to Find First and Last Position of Element in Sorted Array
Mastering Binary Search: Efficiently Find the Starting and Ending Position of a Target Value in a Sorted Array with O(log n) Complexity
The Problem
With this article, I will be covering the Find First and Last Position of an Element in a Sorted Array problem.
Leetcode describes the problem with the following:
Given an array of integers
nums
sorted in non-decreasing order, find the starting and ending position of a giventarget
value.If
target
is not found in the array, return[-1, -1]
.You must write an algorithm with
O(log n)
runtime complexity.
Example:
Input: nums = [5,7,7,8,8,10], target = 8
Output: [3,4]
Leetcode ranks this problem as a medium. I think that is an appropriate rating. The solution is feasible but does require some algorithmic understanding.
Naive Approach and Its Limitations
The naive approach for solving this problem would be to scan through the array linearly to find the first and last occurrence of the target value. This involves looping through the array once to find the first occurrence of the target and marking that index as the starting position, then looping through it again to find the last occurrence and marking that as the ending position.
While this approach works, it takes O(n)
time to solve, which doesn't meet the constraint of O(log n)
runtime complexity. Therefore, it would become inefficient when dealing with large datasets.
The Solution
To achieve a runtime complexity of O(log n)
, we can use binary search. A binary search is possible because the array is already sorted. In this optimized approach, we will perform two binary searches:
Finding the Leftmost Position: The first binary search will find the leftmost or the first occurrence of the target value.
Finding the Rightmost Position: The second binary search will find the rightmost or the last occurrence of the target value.
Here's how each binary search would work:
Leftmost Position: Initialize
left
to 0 andright
ton - 1
(wheren
is the length of the array). In the while loop, calculate the middle index as(left + right) // 2
. Iftarget > nums[mid]
, setleft = mid + 1
. Otherwise, setright = mid - 1
. After the loop, check ifnums[left]
is the target to confirm.Rightmost Position: Initialize
left
to 0 andright
ton - 1
again. This time, iftarget >= nums[mid]
, setleft = mid + 1
. Otherwise, setright = mid - 1
. After the loop, check ifnums[right]
is the target to confirm.Finally return [-1, -1] if the location of the left and right position overlap.
Notice that the key difference between the left and right searches is the use of the greater than or equal to check on the right search.
class Solution(object):
def searchRange(self, nums, target):
left, right = 0, len(nums) - 1
while left <= right:
mid = (left + right) / 2
if target > nums[mid]:
left = mid + 1
else:
right = mid - 1
left_to_return = left
left, right = 0, len(nums) - 1
while left <= right:
mid = (left + right) / 2
if target >= nums[mid]:
left = mid + 1
else:
right = mid - 1
if left_to_return <= right:
return (left_to_return, right)
else:
return [-1, -1]
Time Complexity
Each binary search has a time complexity of O(log n)
, and since we are performing two binary searches, the overall time complexity remains O(log n)
.
This optimized approach not only meets the problem's algorithmic constraint but also efficiently finds the target's starting and ending positions in the array.