二分查找也稱折半查找(Binary Search),它是一種效率較高的查找方法。但是,折半查找要求線性表必須采用順序存儲結構,而且表中元素按關鍵字有序排列。
網站建設哪家好,找創新互聯公司!專注于網頁設計、網站建設、微信開發、微信小程序、集團企業網站建設等服務項目。為回饋新老客戶創新互聯還提供了阿合奇免費建站歡迎大家使用!
二分查找優缺點
優點是比較次數少,查找速度快,平均性能好;
其缺點是要求待查表為有序表,且插入刪除困難。
因此,折半查找方法適用于不經常變動而查找頻繁的有序列表。
使用條件:查找序列是順序結構,有序。
過程
首先,假設表中元素是按升序排列,將表中間位置記錄的關鍵字與查找關鍵字比較,如果兩者相等,則查找成功;否則利用中間位置記錄將表分成前、后兩個子表,如果中間位置記錄的關鍵字大于查找關鍵字,則進一步查找前一子表,否則進一步查找后一子表。重復以上過程,直到找到滿足條件的記錄,使查找成功,或直到子表不存在為止,此時查找不成功。
利用循環的方式實現二分法查找
public class BinarySearch {
public static void main(String[] args) {
// 生成一個隨機數組 ? ? ? ?int[] array = suiji();
// 對隨機數組排序 ? ? ? ?Arrays.sort(array);
System.out.println("產生的隨機數組為: " + Arrays.toString(array));
System.out.println("要進行查找的值: ");
Scanner input = new Scanner(System.in);
// 進行查找的目標值 ? ? ? ?int aim = input.nextInt();
// 使用二分法查找 ? ? ? ?int index = binarySearch(array, aim);
System.out.println("查找的值的索引位置: " + index);
}
/** ? ? * 生成一個隨機數組 ? ? *
* @return 返回值,返回一個隨機數組 ? ? */
private static int[] suiji() {
// random.nextInt(n)+m ?返回m到m+n-1之間的隨機數 ? ? ? ?int n = new Random().nextInt(6) + 5;
int[] array = new int[n];
// 循環遍歷為數組賦值 ? ? ? ?for (int i = 0; i array.length; i++) {
array[i] = new Random().nextInt(100);
}
return array;
}
/** ? ? * 二分法查找 ?---循環的方式實現 ? ? *
* @param array 要查找的數組 ? ? * @param aim 要查找的值 ? ? * @return 返回值,成功返回索引,失敗返回-1 ? ? */
private static int binarySearch(int[] array, int aim) {
// 數組最小索引值 ? ? ? ?int left = 0;
// 數組最大索引值 ? ? ? ?int right = array.length - 1;
int mid;
while (left = right) {
mid = (left + right) / 2;
// 若查找數值比中間值小,則以整個查找范圍的前半部分作為新的查找范圍 ? ? ? ? ? ?if (aim array[mid]) {
right = mid - 1;
// 若查找數值比中間值大,則以整個查找范圍的后半部分作為新的查找范圍 ? ? ? ? ? ?} else if (aim array[mid]) {
left = mid + 1;
// 若查找數據與中間元素值正好相等,則放回中間元素值的索引 ? } else {
return mid;
}
}
return -1;
}}
運行結果演示:
由以上運行結果我們得知,如果要查找的數據在數組中存在,則輸出該數據在數組中的索引;如果不存在則輸出 -1 ,也就是打印 -1 則該數在數組中不存在,反之則存在。
四、利用遞歸的方式實現二分法查找
public class BinarySearch2 {
public static void main(String[] args) {
// 生成一個隨機數組 ? ? ? ?int[] array = suiji();
// 對隨機數組排序 ? ? ? ?Arrays.sort(array);
System.out.println("產生的隨機數組為: " + Arrays.toString(array));
System.out.println("要進行查找的值: ");
Scanner input = new Scanner(System.in);
// 進行查找的目標值 ? ? ? ?int aim = input.nextInt();
// 使用二分法查找 ? ? ? ?int index = binarySearch(array, aim, 0, array.length - 1);
System.out.println("查找的值的索引位置: " + index);
}
/** ? ? * 生成一個隨機數組 ? ? * ? ? * @return 返回值,返回一個隨機數組 ? ? */
private static int[] suiji() {
// Random.nextInt(n)+m ?返回m到m+n-1之間的隨機數 ? ? ? ?int n = new Random().nextInt(6) + 5;
int[] array = new int[n];
// 循環遍歷為數組賦值 ? ? ? ?for (int i = 0; i array.length; i++) {
array[i] = new Random().nextInt(100);
}
return array;
}
/** ? ? * 二分法查找 ---遞歸的方式 ? ? * ? ? * @param array 要查找的數組 ? ? * @param aim ? 要查找的值 ? ? * @param left ?左邊最小值 ? ? * @param right 右邊最大值 ? ? * @return 返回值,成功返回索引,失敗返回-1 ? ? */
private static int binarySearch(int[] array, int aim, int left, int right) {
if (aim array[left] || aim array[right]) {
return -1;
}
// 找中間值 ? ? ? ?int mid = (left + right) / 2;
if (array[mid] == aim) {
return mid;
} else if (array[mid] aim) {
//如果中間值大于要找的值則從左邊一半繼續遞歸 ? ? ? ? ? ?return binarySearch(array, aim, left, mid - 1);
} else {
//如果中間值小于要找的值則從右邊一半繼續遞歸 ? ? ? ? ? ?return binarySearch(array, aim, mid + 1, array.length-1);
}
}}
運行結果演示:
總結:
遞歸相較于循環,代碼比較簡潔,但是時間和空間消耗比較大,效率低。在實際的學習與工作中,根據情況選擇使用。通常我們如果使用循環實現代碼只要不是太繁瑣都選擇循環的方式實現~
import java.util.Scanner;
public class Test {
static int bsearch( int[] a, int v ) {
int l, r;
l = 0; r = a.length-1;
while ( l = r ) {
int m = (l+r)/2;
if ( a[m] == v ) return m; else
if ( a[m] v ) r = m-1; else
if ( a[m] v ) l = m+1;
}
return -1;
}
public static void main( String[] args ) {
int[] a = { 1,3,5,7,9 };
Scanner sc = new Scanner(System.in);
System.out.println("請輸入您要找的值:");
int num = sc.nextInt();
System.out.println( "找到 " + num + " 在數組的位置是:" + bsearch( a, num ) );
}
}
二分法查找(折半查找)的時間復雜度是O(log2n)
即是最壞的情況比較次數是2為底2n的對數。也就數如果數組長度為2,最壞的情況比較2兩次;數組長度為16,最壞的情況比較5次;數組長度1204,最壞的情況是比較11次 就可以找到這個值或者確定找不到這個值。
你的代碼就是通過判斷比較的次數來決定是否結束循環,當已比較(循環)次數大于最壞情況的次數還沒有結束(number != a[middle]),則說明數組中不存在這個值。不過這里是用的N/2來近似的判斷。
另一種更普遍的寫法
public?class?Demo?{
public?static?void?main(String[]?args)?{
//?你原來的代碼
System.out.println(Arrays.toString(a));
Scanner?scanner?=?new?Scanner(System.in);
System.out.println("輸入整數,程序判斷該整數是否在數組中:");
int?number?=?scanner.nextInt();
int?index?=?binary(a,?number);
if?(index?==?-1)?{
System.out.printf("%d不在數組中.\n",?number);
}?else?{
System.out.printf("%d在數組中,?在數組中的位置下標是%d.",?number,?index);
}
}
private?static?int?binary(int[]?array,?int?value)?{
int?start?=?0;
int?end?=?array.length?-?1;
while?(start?=?end)?{
int?middle?=?(start?+?end)?/?2;
if?(value?==?array[middle])?{
return?middle;
}?else?if?(value??array[middle])?{
start?=?middle?+?1;
}?else?{
end?=?middle?-?1;
}
}
return?-1;
}
}
網站題目:java折半查找代碼實現 java折半查找算法
當前路徑:http://vcdvsql.cn/article14/ddojpge.html
成都網站建設公司_創新互聯,為您提供企業網站制作、軟件開發、網頁設計公司、網站排名、外貿建站、App開發
聲明:本網站發布的內容(圖片、視頻和文字)以用戶投稿、用戶轉載內容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網站立場,如需處理請聯系客服。電話:028-86922220;郵箱:631063699@qq.com。內容未經允許不得轉載,或轉載時需注明來源: 創新互聯