【谷歌面試題】給出一個陣列A,找出一對 (i, j)使得A[i]
題目:給出一個陣列A,找出一對 (i, j)使得A[i] <= A[j] (i <= j)並且j-i最大 ,若有多個這樣的位置對,返回i最小的那一對。
最直接的想法就是對於每一個 i 從陣列最尾端開始向前找到第一個大於等於 A[i] 的位置 j ,時間複雜度O(n^2)。
pair<int, int> find(const vector<int> &A)
{
int n = A.size();
if(n == 0)
throw new invalid_argument("Array's size can't be 0!");
int target_i = 0, target_j = 0;
int max_len = 0;
for(int i = 0; i < n; ++i)
{
int j;
for(j = n-1; j >= i; --j)
if(A[j] >= A[i])
break;
if(j-i+1 > max_len)
{
target_i = i;
target_j = j;
max_len = j-i+1;
}
}
return make_pair<int, int>(target_i, target_j);
}
我們對上述演算法稍作優化。當i=0時,我們假設找到的大於A[i]的最右位置是j0,那麼對於i=1時,我們根本就不需要考慮小於j0的位置,因為它們的區間長度都小於j0+1,不可能成為最優解。pair<int, int> find(const vector<int> &A)
{
int n = A.size();
if(n == 0)
throw new invalid_argument("Array's size can't be 0!");
int target_i = 0, target_j = 0;
int max_len = 0;
for(int i = 0; i < n; ++i)
{
int j;
for(j = n-1; j > target_j; --j) // 此處只需檢查到target_j
if(A[j] >= A[i])
break;
if(j-i+1 > max_len)
{
target_i = i;
target_j = j;
max_len = j-i+1;
}
}
return make_pair<int, int>(target_i, target_j);
}
但時間複雜度仍然是O(n^2)的。我們可以繼續接著上面的思路優化。其實對於位置 i 求最後一個大於等於它的位置,不需要每次都從陣列尾部向前找,我們可以通過改進這個地方將時間複雜度變為O(n)。
過程是這樣的,對於 i ,我們先找到 i 及其右端的最大元素的位置 j ,檢查是否比當前記錄的最優解更優,更新。然後考慮 j+1及其右端的最大元素位置是否大於等於A[i],若是,令 j 等於該位置,重複如上過程,若否,那麼從位置i+1重新開始,但j仍然從當前位置考慮即可,原因上面已說明。這樣時間複雜度就成O(n)的了。
具體請參考程式碼
pair<int, int> find(const vector<int> &A)
{
int n = A.size();
if(n == 0)
throw new invalid_argument("Array's size can't be 0!");
vector<int> right_max_pos(n);
right_max_pos[n-1] = n-1;
for(int i = n-2; i >= 0; --i)
{
if(A[i] > A[right_max_pos[i+1]])
right_max_pos[i] = i;
else
right_max_pos[i] = right_max_pos[i+1];
}
int max_len = 0;
int target_i, target_j;
int i = 0, j = 0;
while(j < n)
{
j = right_max_pos[j];
if(A[j] >= A[i])
{
if(j-i+1 > max_len)
{
target_i = i;
target_j = j;
max_len = j-i+1;
}
++j;
}
else
++i;
}
return make_pair<int, int>(target_i, target_j);
}
這裡簡單說一下測試方法,測試我們可以先測試最簡單的實現方案,這裡的第一種實現,因為這種實現簡單,出現錯誤的可能性小,測試起來簡單。測試時可以不考慮時間複雜度,只考慮正確性。然後我們使用此經過測試過的演算法的輸入輸出去測試其他演算法(對比結果)。
相關文章
- 面試題:給定陣列a,找到最大的j-i, 使a[j]>a[i]面試題陣列
- 前端面試題,寫出一個快速找出兩個陣列不同值的函式。前端面試題陣列函式
- 每日一題(一):找出兩個陣列中共有的數字每日一題陣列
- 將一個陣列賦值給另外一個陣列陣列賦值
- 每日一題 - 劍指 Offer 53 - I. 在排序陣列中查詢數字 I每日一題排序陣列
- 一個關於 i++ 和 ++i 的面試題打趴了所有人面試題
- FCC-JS-BAS-Where do I belong-陣列排序並找出元素索引JS陣列排序索引
- 給定一個整數陣列,找出總和最大的連續數列,並返回總和。陣列
- 陣列的一些面試題陣列面試題
- 找出陣列中只出現一次的數字陣列
- 計算給定整數陣列中,第i個元素表示從第i個位置開始按摩到最後一個位置能夠獲得的最大收益。陣列
- 給出分數陣列,得到對應的名次陣列陣列
- 記一次面試一直出現的題——陣列去重面試陣列
- js練習(一)找出元素item在給定陣列arr中的位置JS陣列
- 一道遞迴陣列面試題遞迴陣列面試題
- 【谷歌面試題】求陣列中兩個元素的最小距離谷歌面試題陣列
- 針對一個陣列的排序,面試官會這樣問陣列排序面試
- 三角形個數:小b有一個僅包含非負整數的陣列a,她想知道有多少個三元組(i,j,k),滿足i<j<k且a[i],a[j],a[k]可能作為某個三角形的三條邊的邊長。陣列
- 找出陣列中元素出現次數超過陣列長度一半的元素陣列
- 給定一個非空整數陣列,除了某個元素只出現一次以外,其餘每個元素均出現兩次。找出那個只出現了一次的元素。陣列
- 兩個陣列分別取出一個來相加,找出和最小的k個陣列
- 在陣列中找出唯一的重複元素陣列
- 給定一個整數陣列 nums 和一個目標值 target,請你在該陣列中找出和為目標值的那 兩個 整數,並返回他們的陣列下標。陣列
- 給定一個整數陣列和一個目標值,找出陣列中和為目標值的兩個數。 你可以假設每個輸入只對應一種答案,且同樣的元素不能被重複利用陣列
- 【筆記】9i 文件中的一個問題筆記
- 如何將一個陣列中的元素插入另一個陣列陣列
- golang 陣列面試題Golang陣列面試題
- 一個陣列和另一個陣列比較,如果第一個陣列裡面有的就不新增了陣列
- Single Number 陣列中除了某個元素出現一次,其他都出現兩次,找出這個元素陣列
- 陣列1——求一個陣列的最大子陣列陣列
- 改進,從一個陣列中找出 N 個數,其和為 M 的所有可能陣列
- 【每日一題】快照陣列每日一題陣列
- 劍指 Offer 56 - I. 陣列中數字出現的次數陣列
- 將一個陣列複製到另一個陣列上陣列
- 小米借谷歌I/O釋出美國版小米盒子谷歌
- python實現給定一個數和陣列,求陣列中兩數之和為給定的數Python陣列
- 給出一個由[-100,100]之間整陣列成的陣列,求其相加和最大的連續子陣列 輸入 一個連續整陣列成的陣列 輸出 子陣列相加的最大值 樣例輸入 -......陣列
- javascript將陣列的元素每兩個一組存入一個新陣列JavaScript陣列