2014-11-15 130 views
0

我有一個排序數組,並希望遞歸地找到插入元素的位置。例如。插入值爲3的數組2,4,5,6,7,8,12,35,算法應該返回3必須插入的位置的索引(索引1)。 下面的代碼有時會起作用,有時候它會陷入無限循環。最終,我的大腦感覺像果凍,我要求你的幫助。這是代碼:遞歸插入通過二進制搜索

private static int insertRec(int[] a, int e, int l, int r) { 
    int mid = l + r/2; 
    if (l == r || a[mid] == e) return mid; 
    if (e > a[mid]) insertRec(a, e, mid + 1, r); 
    if (e < a[mid]) insertRec(a, e, l, mid - 1); 
    return -1; 

} 

編輯現出工作代碼:

private static int insertRec(int[] a, int e, int l, int r) { 
    int mid = (l + r)/2; 
    if (l == r || a[mid] == e) return mid; 
    else if (e > a[mid]) return insertRec(a, e, mid + 1, r); 
    else if (e < a[mid]) return insertRec(a, e, l, mid - 1); 
    return -1; 

} 

回答

3
int mid = l + r/2; 

應該

int mid = (l + r)/2; 

編輯:此外,你可以檢查的說明STL算法lower_bound,它正是你想要的,就像我一樣erstand。基於這方面的一個實現是這樣的:

int lb(int a[], int last, int val) { // array, len of the array, element to insert 
    int it, count, step; 
    int first = 0; 
    count = (last-first); 
    while (count > 0) { 
    it = first; 
    step = count/2; 
    it += step; 
    if (a[it] < val) { 
     first = ++it; 
     count -= step+1; 
    } else 
     count = step; 
    } 
    return first; 
} 

EDIT2:你的代碼有一些錯誤的正常工作,其中包括:在第二行

  • 你無法檢查a[mid] == e因爲您要插入的元素可能不存在於數組中。這會導致在幾種情況下返回-1。

  • 無限循環產生於計算mid並稍後分配mid+1mid-1的方式。

  • ,因爲你要插入的元素可以等於陣列中的一些元素,你會被錯誤地回來,因爲你比較兩個​​和e < a[mid]

我建議你看看這個奇妙的post關於二進制搜索。無論如何,下面的算法試圖追蹤更多可能的風格和使用帖子的信息。希望能幫助到你。

private static int insertRec(int[] a, int e, int l, int r) { 
    int mid = l + (r - l)/2; 
    if (l == r) return l; 
    else if (a[mid] < e) return insertRec(a, e, mid+1, r); 
    else return insertRec(a, e, l, mid); 
} 
+0

你是完全正確的關於這個(我提到我的果凍的大腦?) 但是,這並不是我的問題在整個解決方案。取數組0,0,0,1,2,2,3,5,7,7,並嘗試用我的算法插入3。它會產生-1 – gutenmorgenuhu

+0

@gutenmorgenuhu檢查我的編輯。我沒有完全檢查你的算法,只是顯而易見的部分。 – ale64bit

+0

我感謝你的快速回答。問題是,你的lb算法不使用二分搜索,而是執行線性迭代。 – gutenmorgenuhu