C++實(shí)現(xiàn)LeetCode(34.在有序數(shù)組中查找元素的第一個(gè)和最后一個(gè)位置)
[LeetCode] 34. Find First and Last Position of Element in Sorted Array 在有序數(shù)組中查找元素的第一個(gè)和最后一個(gè)位置
Given an array of integers nums sorted in ascending order, find the starting and ending position of a given target value.
Your algorithm's runtime complexity must be in the order of O(log n).
If the target is not found in the array, return [-1, -1].
Example 1:
Input: nums = [5,7,7,8,8,10], target = 8
Output: [3,4]
Example 2:
Input: nums = [5,7,7,8,8,10], target = 6
Output: [-1,-1]
這道題讓我們?cè)谝粋€(gè)有序整數(shù)數(shù)組中尋找相同目標(biāo)值的起始和結(jié)束位置,而且限定了時(shí)間復(fù)雜度為 O(logn),這是典型的二分查找法的時(shí)間復(fù)雜度,所以這里也需要用此方法,思路是首先對(duì)原數(shù)組使用二分查找法,找出其中一個(gè)目標(biāo)值的位置,然后向兩邊搜索找出起始和結(jié)束的位置,代碼如下:
解法一:
class Solution {
public:
vector<int> searchRange(vector<int>& nums, int target) {
int idx = search(nums, 0, nums.size() - 1, target);
if (idx == -1) return {-1, -1};
int left = idx, right = idx;
while (left > 0 && nums[left - 1] == nums[idx]) --left;
while (right < nums.size() - 1 && nums[right + 1] == nums[idx]) ++right;
return {left, right};
}
int search(vector<int>& nums, int left, int right, int target) {
if (left > right) return -1;
int mid = left + (right - left) / 2;
if (nums[mid] == target) return mid;
if (nums[mid] < target) return search(nums, mid + 1, right, target);
else return search(nums, left, mid - 1, target);
}
};
可能有些人會(huì)覺(jué)得上面的算法不是嚴(yán)格意義上的 O(logn) 的算法,因?yàn)樵谧顗牡那闆r下會(huì)變成 O(n),比如當(dāng)數(shù)組里的數(shù)全是目標(biāo)值的話,從中間向兩邊找邊界就會(huì)一直遍歷完整個(gè)數(shù)組,那么下面來(lái)看一種真正意義上的 O(logn) 的算法,使用兩次二分查找法,第一次找到左邊界,第二次調(diào)用找到右邊界即可,具體代碼如下:
解法二:
class Solution {
public:
vector<int> searchRange(vector<int>& nums, int target) {
vector<int> res(2, -1);
int left = 0, right = nums.size();
while (left < right) {
int mid = left + (right - left) / 2;
if (nums[mid] < target) left = mid + 1;
else right = mid;
}
if (right == nums.size() || nums[right] != target) return res;
res[0] = right;
right = nums.size();
while (left < right) {
int mid = left + (right - left) / 2;
if (nums[mid] <= target) left = mid + 1;
else right = mid;
}
res[1] = right - 1;
return res;
}
};
其實(shí)我們也可以只使用一個(gè)二分查找的子函數(shù),來(lái)同時(shí)查找出第一個(gè)和最后一個(gè)位置。如何只用查找第一個(gè)大于等于目標(biāo)值的二分函數(shù)來(lái)查找整個(gè)范圍呢,這里用到了一個(gè)小 trick,首先來(lái)查找起始位置的 target,就是在數(shù)組中查找第一個(gè)大于等于 target 的位置,當(dāng)返回的位置越界,或者該位置上的值不等于 target 時(shí),表示數(shù)組中沒(méi)有 target,直接返回 {-1, -1} 即可。若查找到了 target 值,則再查找第一個(gè)大于等于 target+1 的位置,然后把返回的位置減1,就是 target 的最后一個(gè)位置,即便是返回的值越界了,減1后也不會(huì)越界,這樣就實(shí)現(xiàn)了使用一個(gè)二分查找函數(shù)來(lái)解題啦,參見(jiàn)代碼如下:
解法三:
class Solution {
public:
vector<int> searchRange(vector<int>& nums, int target) {
int start = firstGreaterEqual(nums, target);
if (start == nums.size() || nums[start] != target) return {-1, -1};
return {start, firstGreaterEqual(nums, target + 1) - 1};
}
int firstGreaterEqual(vector<int>& nums, int target) {
int left = 0, right = nums.size();
while (left < right) {
int mid = left + (right - left) / 2;
if (nums[mid] < target) left = mid + 1;
else right = mid;
}
return right;
}
};
到此這篇關(guān)于C++實(shí)現(xiàn)LeetCode(34.在有序數(shù)組中查找元素的第一個(gè)和最后一個(gè)位置)的文章就介紹到這了,更多相關(guān)C++實(shí)現(xiàn)在有序數(shù)組中查找元素的第一個(gè)和最后一個(gè)位置內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C++實(shí)現(xiàn)LeetCode(38.計(jì)數(shù)和讀法)
- C++實(shí)現(xiàn)LeetCode(51.N皇后問(wèn)題)
- C++實(shí)現(xiàn)LeetCode(77.Combinations 組合項(xiàng))
- C++實(shí)現(xiàn)LeetCode(46.全排列)
- C++實(shí)現(xiàn)LeetCode(37.求解數(shù)獨(dú))
- C++實(shí)現(xiàn)LeetCode(36.驗(yàn)證數(shù)獨(dú))
- C++實(shí)現(xiàn)LeetCode(35.搜索插入位置)
- C++實(shí)現(xiàn)LeetCode(33.在旋轉(zhuǎn)有序數(shù)組中搜索)
- C++實(shí)現(xiàn)LeetCode(39.組合之和)
相關(guān)文章
C/C++使用Zlib實(shí)現(xiàn)文件的壓縮與解壓
zlib 是一個(gè)開(kāi)源的數(shù)據(jù)壓縮庫(kù),旨在提供高效、輕量級(jí)的壓縮和解壓縮算法,本文將介紹如何使用 zlib 庫(kù)進(jìn)行數(shù)據(jù)的壓縮和解壓縮,以及如何保存和讀取壓縮后的文件,感興趣的可以了解下2023-11-11
內(nèi)聯(lián)函數(shù)inline與宏定義深入解析
類的內(nèi)斂函數(shù)是一個(gè)真正的函數(shù)。使用內(nèi)聯(lián)函數(shù)inline可以完全取代表達(dá)式形式的宏定義2013-09-09
???????C語(yǔ)言實(shí)現(xiàn)單鏈表基本操作方法
這篇文章主要介紹了???????C語(yǔ)言實(shí)現(xiàn)單鏈表基本操作方法,文章圍繞主題展開(kāi)詳細(xì)介紹,具有一定的參考價(jià)值,需要的小伙伴可以參考一下2022-05-05
C語(yǔ)言實(shí)現(xiàn)數(shù)字連連看
這篇文章主要為大家詳細(xì)介紹了C語(yǔ)言實(shí)現(xiàn)數(shù)字連連看游戲,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-09-09
C++實(shí)現(xiàn)飛機(jī)大戰(zhàn)
這篇文章主要為大家詳細(xì)介紹了C++實(shí)現(xiàn)飛機(jī)大戰(zhàn),文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2020-11-11
c++ vector對(duì)象相關(guān)總結(jié)
這篇文章主要介紹了c++ vector對(duì)象的相關(guān)資料,幫助大家更好的理解和學(xué)習(xí)使用c++,感興趣的朋友可以了解下2021-02-02

