這篇文章來講二分法,這是一種在實際情況中十分常用的算法
成都創新互聯公司專業為企業提供正安網站建設、正安做網站、正安網站設計、正安網站制作等企業網站建設、網頁設計與制作、正安企業網站模板建站服務,10余年正安做網站經驗,不只是建網站,更提供有價值的思路和整體網絡服務。我們之前講過,解決計算機問題的一個常規方案就是暴力搜索,即:遍歷整個搜索空間,找到給定問題的解
在這個基礎上,針對問題的不同特征,我們可以應用不同的數據結構和算法,去優化搜索的時間和空間效率
二分搜索算法就是針對有序區間的元素搜索問題進行的時間效率優化
換句話說,區間有序是應用二分搜索算法的必要條件
如果要求在亂序數組中找到給定值,那么唯一的做法就是逐個遍歷數組元素
如果要求在有序數組中找到給定值,那么就可以使用二分搜索算法進行處理
二分搜索算法的思路很簡單:通過比較區間中值和給定值,每次可以縮小一半搜索區間
舉個例子,給定有序數組[1, 2, 3, 4, 5, 6, 7, 8, 9]
,要求找出元素6
,算法步驟如下:
5
和給定值6
,發現5< 6
,所以給定值必在區間右半部分[6, 7, 8, 9]
7
和給定值6
,發現7 >6
,所以給定值比在區間左半部分[6]
6
和給定值6
,發現6 = 6
,至此就能找出給定值二分搜索算法的思路很簡單,但實現起來需要注意的細節卻有很多
下面先按上述所說思路給出一個代碼模板,該模板用于在升序數組中查找給定值
如果能夠找到,則返回對應下標;如果沒有找到,則返回-1
int binary_search(vector& nums, int target) {int n = nums.size();
// 定義搜索區間為 [p1 , p2]
int p1 = 0;
int p2 = n - 1;
// 定義終止條件為 (p1 >p2)
while (p1<= p2) {// 計算中值:
// 一般計算中值的方式是 (p1 + p2) / 2
// 為了防止相加溢出改成 ?
int mid = p1 + (p2 - p1) / 2;
// 分類討論:
// 若中值等于給定值,則表示已找到,返回結果就好
// 若中值小于給定值,則將搜索區間約束在右半部分
// 若中值大于給定值,則將搜索區間約束在左半部分
if (nums[mid] == target) {// 中值等于給定值
// 返回結果
return mid;
} else if (nums[mid]< target) {// 中值小于給定值
// 更新搜索區間為右半部分,此時左邊界更新,右邊界不變
p1 = mid + 1;
} else if (nums[mid] >target) {// 中值大于給定值
// 更新搜索區間為左半部分,此時右邊界更新,左邊界不變
p2 = mid - 1;
}
}
// 沒有找到,返回結果
return -1;
}
上述的代碼思路很清晰,但實際寫起來可能會有很多細節值得注意
常見的問題集中在:搜索區間的定義、終止條件的定義、搜索區間的更新
搜索區間的定義,第4
、5
行
這里定義的搜索區間是左閉右閉區間[p1, p2]
,初始化為p1 = 0, p2 = n - 1
當然也有其他人定義為左閉右開區間[p1, p2)
,初始化為p1 = 0, p2 = n
這兩種定義方式都是沒有問題的,區別在于后續要怎么定義終止條件和更新搜索區間
終止條件的定義,第7
行
當搜索區間為空時,就可以終止搜索,具體來說:
對于[p1, p2]
,區間為空時有p1 >p2
,也即while
運行條件應為p1<= p2
對于[p1, p2)
,區間為空時有p1 >= p2
,也即while
運行條件應為p1< p2
搜索區間的更新,第23
、27
行
判斷完中值和給定值的大小關系后,新的搜索區間應該去掉中值分為左右兩部分,具體來說:
對于[p1, p2]
,新的搜索區間應為[p1, mid - 1]
和[mid + 1, p2]
對于[p1, p2)
,新的搜索區間應為[p1, mid)
和[mid + 1, p2)
下面針對三種常見的二分搜索場景給出代碼模板,以后遇到相似的場景時可以舉一反三地解決問題
在升序數組中查找給定值唯一出現的位置,降序數組思路類似,可以自行推理
如果能夠找到,則返回對應下標;如果沒有找到,則返回-1
int binary_search(vector& nums, int target) {int n = nums.size();
int p1 = 0;
int p2 = n - 1;
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
if (nums[mid] == target) {// 若中值等于給定值,則表示已找到,返回結果就好
return mid;
} else if (nums[mid]< target) {// 若中值小于給定值,則將搜索區間約束在右半部分
p1 = mid + 1;
} else if (nums[mid] >target) {// 若中值大于給定值,則將搜索區間約束在左半部分
p2 = mid - 1;
}
}
return -1;
}
在升序數組中查找給定值最先出現的位置,降序數組思路類似,可以自行推理
如果能夠找到,則返回對應下標;如果沒有找到,則返回-1
int lower_bound(vector& nums, int target) {int n = nums.size();
int p1 = 0;
int p2 = n - 1;
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
if (nums[mid] == target) {// 不同之處
// 若中值等于給定值,則將搜索范圍約束在左半部分繼續搜索
p2 = mid - 1;
} else if (nums[mid]< target) {// 若中值小于給定值,則將搜索區間約束在右半部分
p1 = mid + 1;
} else if (nums[mid] >target) {// 若中值大于給定值,則將搜索區間約束在左半部分
p2 = mid - 1;
}
}
// 不同之處
// 最后檢查 p1 是否符合條件
if (p1 >n - 1 || nums[p1] != target) {return -1;
}
return p1;
}
在升序數組中查找給定值最后出現的位置,降序數組思路類似,可以自行推理
如果能夠找到,則返回對應下標;如果沒有找到,則返回-1
int upper_bound(vector& nums, int target) {int n = nums.size();
int p1 = 0;
int p2 = n - 1;
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
if (nums[mid] == target) {// 不同之處
// 若中值等于給定值,則將搜索范圍約束在右半部分繼續搜索
p1 = mid + 1;
} else if (nums[mid]< target) {// 若中值小于給定值,則將搜索區間約束在右半部分
p1 = mid + 1;
} else if (nums[mid] >target) {// 若中值大于給定值,則將搜索區間約束在左半部分
p2 = mid - 1;
}
}
// 不同之處
// 最后檢查 p2 是否符合條件
if (p2< 0 || nums[p2] != target) {return -1;
}
return p2;
}
(1)在有序數組中查找給定值可插入位置 | leetcode35
給定一個已排序數組和一個目標值,不考慮重復元素
如果目標值在數組內,返回其索引,如果目標值不在數組內,返回其按順序插入的位置
解題思路,可轉換為在有序數組中查找第一個大于等于給定值的位置
class Solution {public:
int searchInsert(vector& nums, int target) {int n = nums.size();
int p1 = 0;
int p2 = n - 1;
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
if (nums[mid] == target) {p2 = mid - 1;
} else if (nums[mid]< target) {p1 = mid + 1;
} else if (nums[mid] >target) {p2 = mid - 1;
}
}
return p1;
}
};
(2)在有序數組中查找給定值出現的范圍 | leetcode34
給定一個已排序數組和一個目標值,數組中有重復的元素
找出目標值在數組中的開始位置和結束位置,如果目標值不在數組內,則返回[-1, -1]
解題思路,可轉換為在有序數組中查找給定值最先出現的位置和在有序數組中查找給定值最后出現的位置
class Solution {public:
vectorsearchRange(vector& nums, int target) {int p1 = lower_bound(nums, target);
int p2 = upper_bound(nums, target);
return (p1 == -1 || p2 == -1) ? vector{-1, -1} : vector{p1, p2};
}
int lower_bound(vector& nums, int target) {int n = nums.size();
int p1 = 0;
int p2 = n - 1;
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
if (nums[mid] == target) {p2 = mid - 1;
} else if (nums[mid]< target) {p1 = mid + 1;
} else if (nums[mid] >target) {p2 = mid - 1;
}
}
if (p1 >n - 1 || nums[p1] != target) {return -1;
}
return p1;
}
int upper_bound(vector& nums, int target) {int n = nums.size();
int p1 = 0;
int p2 = n - 1;
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
if (nums[mid] == target) {p1 = mid + 1;
} else if (nums[mid]< target) {p1 = mid + 1;
} else if (nums[mid] >target) {p2 = mid - 1;
}
}
if (p2< 0 || nums[p2] != target) {return -1;
}
return p2;
}
};
(3)吃香蕉 | leetcode875
給定一個數組,數組中的元素piles[i]
表示第i
堆香蕉的數量,單位為根
返回能在給定時間h
內吃完所有香蕉的最小速度k
,其中h
的單位為時,k
的單位為根/時
在每小時中,只能選擇一堆香蕉,如果這堆香蕉小于k
根,吃完后這小時也不能吃別的香蕉
解題思路
- 吃香蕉的速度和能否在給定時間內吃完所有香蕉存在單調性,可以考慮用二分查找搜索最小速度
- 另一個問題是給定一個速度,怎么計算需要多少時間才能吃完所有香蕉
class Solution {public:
int minEatingSpeed(vector& piles, int h) {// 左邊界為最小速度,顯然為一
// 右邊界為大速度,因為每小時最多只能吃完一堆香蕉,所以取每堆香蕉中的大根數即可
int p1 = 1;
int p2 = 0;
for (int pile: piles) {p2 = max(p2, pile);
}
// 二分查找
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
long long int tmp = getTime(piles, mid);
if (tmp == h) {p2 = mid - 1;
} else if (tmp< h) {p2 = mid - 1;
} else if (tmp >h) {p1 = mid + 1;
}
}
return p1;
}
// 給定一個速度
// 計算需要多少時間才能吃完所有香蕉
long long int getTime(vector& piles, int speed) {long long int time = 0;
for (int pile: piles) {// (a + b - 1) / b 相當于 a / b 向上取整
time += (pile + speed - 1) / speed;
}
return time;
}
};
(6)運包裹 | leetcode1011
給定一個數組,數組中的元素weights[i]
表示第i
個包裹的重量,單位為噸
返回能在給定時間d
內運走所有包裹的最小運力c
,其中d
的單位為天,c
的單位為噸/天
在每一天中,需要按數組順序運走若干個包裹,要求運走包裹的重量不能超過c
噸
解題思路
- 運包裹的運力和能否在給定時間內運走所有貨物存在單調性,可以考慮用二分查找搜索最小運力
- 另一個問題是給定一個運力,怎么計算需要多少時間才能運走所有包裹
class Solution {public:
int shipWithinDays(vector& weights, int d) {// 左邊界為最小運力,至少要等于每個包裹中的大重量
// 右邊界為大運力,因為可以一次運走所有包裹,所以取所有包裹重量的總和
int p1 = 0;
int p2 = 0;
for (int weight: weights) {p1 = max(p1, weight);
p2 = p2 + weight;
}
// 二分查找
while (p1<= p2) {int mid = p1 + (p2 - p1) / 2;
int tmp = getTime(weights, mid);
if (tmp == d) {p2 = mid - 1;
} else if (tmp< d) {p2 = mid - 1;
} else if (tmp >d) {p1 = mid + 1;
}
}
return p1;
}
// 給定一個運力
// 計算需要多少時間才能運走所有包裹
int getTime(vector& weights, int capacity) {int curr = 0;
int need = 1;
for (int weight: weights) {if (curr + weight >capacity) {curr = 0;
need = need + 1;
}
curr = curr + weight;
}
return need;
}
};
你是否還在尋找穩定的海外服務器提供商?創新互聯www.cdcxhl.cn海外機房具備T級流量清洗系統配攻擊溯源,準確流量調度確保服務器高可用性,企業級服務器適合批量采購,新人活動首月15元起,快前往官網查看詳情吧
新聞標題:數據結構與算法(七)二分法-創新互聯
轉載注明:http://vcdvsql.cn/article4/dshcoe.html
成都網站建設公司_創新互聯,為您提供網站制作、Google、搜索引擎優化、建站公司、App設計、軟件開發
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯