亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

面試題:Java 實現查找旋轉數組的最小數字

2024-07-14 08:41:27
字體:
來源:轉載
供稿:網友

在算法面試中,面試官總是喜歡圍繞鏈表、排序、二叉樹、二分查找來做文章,而大多數人都可以跟著專業的書籍來做到倒背如流。而面試官并不希望招收的是一位記憶功底很好,但不會活學活用的程序員。所以學會數學建模和分析問題,并用合理的算法或數據結構來解決問題相當重要。

面試題:打印出旋轉數組的最小數字

題目:把一個數組最開始的若干個元素搬到數組的末尾,我們稱之為數組的旋轉。輸入一個遞增排序的數組的一個旋轉,輸出旋轉數組的最小元素。例如數組 {3,4,5,1,2} 為數組 {1,2,3,4,5} 的一個旋轉,該數組的最小值為 1。

要想實現這個需求很簡單,我們只需要遍歷一遍數組,找到最小的值后直接退出循環。代碼實現如下:

public class Test08 {  public static int getTheMin(int nums[]) {    if (nums == null || nums.length == 0) {      throw new RuntimeException("input error!");    }    int result = nums[0];    for (int i = 0; i < nums.length - 1; i++) {      if (nums[i + 1] < nums[i]) {        result = nums[i + 1];        break;      }    }    return result;  }  public static void main(String[] args) {    // 典型輸入,單調升序的數組的一個旋轉    int[] array1 = {3, 4, 5, 1, 2};    System.out.println(getTheMin(array1));    // 有重復數字,并且重復的數字剛好的最小的數字    int[] array2 = {3, 4, 5, 1, 1, 2};    System.out.println(getTheMin(array2));    // 有重復數字,但重復的數字不是第一個數字和最后一個數字    int[] array3 = {3, 4, 5, 1, 2, 2};    System.out.println(getTheMin(array3));    // 有重復的數字,并且重復的數字剛好是第一個數字和最后一個數字    int[] array4 = {1, 0, 1, 1, 1};    System.out.println(getTheMin(array4));    // 單調升序數組,旋轉0個元素,也就是單調升序數組本身    int[] array5 = {1, 2, 3, 4, 5};    System.out.println(getTheMin(array5));    // 數組中只有一個數字    int[] array6 = {2};    System.out.println(getTheMin(array6));    // 數組中數字都相同    int[] array7 = {1, 1, 1, 1, 1, 1, 1};    System.out.println(getTheMin(array7));  }}

打印結果沒什么毛病。不過這樣的方法顯然不是最優的,我們看看有沒有辦法找出更加優質的方法處理。

有序,還要查找?

找到這兩個關鍵字,我們不免會想到我們的二分查找法,但不少小伙伴肯定會問,我們這個數組旋轉后已經不是一個真正的有序數組了,不過倒像是兩個遞增的數組組合而成的,我們可以這樣思考。

我們可以設定兩個下標 low 和 high,并設定 mid = (low + high)/2,我們自然就可以找到數組中間的元素 array[mid],如果中間的元素位于前面的遞增數組,那么它應該大于或者等于 low 下標對應的元素,此時數組中最小的元素應該位于該元素的后面,我們可以把 low 下標指向該中間元素,這樣可以縮小查找的范圍。

同樣,如果中間元素位于后面的遞增子數組,那么它應該小于或者等于 high 下標對應的元素。此時該數組中最小的元素應該位于該中間元素的前面。我們就可以把 high 下標更新到中位數的下標,這樣也可以縮小查找的范圍,移動之后的 high 下標對應的元素仍然在后面的遞增子數組中。

不管是更新 low 還是 high,我們的查找范圍都會縮小為原來的一半,接下來我們再用更新的下標去重復新一輪的查找。直到最后兩個下標相鄰,也就是我們的循環結束條件。

說了一堆,似乎已經繞的云里霧里了,我們不妨就拿題干中的這個輸入來模擬驗證一下我們的算法。

  1. input:{3,4,5,1,2}
  2. 此時 low = 0,high = 4,mid = 2,對應的值分別是:num[low] = 3,num[high] = 2,num[mid] = 5
  3. 由于 num[mid] > num[low],所以 num[mid] 應該是在左邊的遞增子數組中。
  4. 更新 low = mid = 2,num[low] = 5,mid = (low+high)/2 = 3,num[mid] = 1;
  5. high - low ≠ 1 ,繼續更新
  6. 由于 num[mid] < num[high],所以斷定 num[mid] = 1 位于右邊的自增子數組中;
  7. 更新 high = mid = 3,由于 high - mid = 1,所以結束循環,得到最小值 num[high] = 1;

我們再來看看 Java 中如何用代碼實現這個思路:

public class Test08 {  public static int getTheMin(int nums[]) {    if (nums == null || nums.length == 0) {      throw new RuntimeException("input error!");    }    // 如果只有一個元素,直接返回    if (nums.length == 1)      return nums[0];    int result = nums[0];    int low = 0, high = nums.length - 1;    int mid;    // 確保 low 下標對應的值在左邊的遞增子數組,high 對應的值在右邊遞增子數組    while (nums[low] >= nums[high]) {      // 確保循環結束條件      if (high - low == 1) {        return nums[high];      }      // 取中間位置      mid = (low + high) / 2;      // 代表中間元素在左邊遞增子數組      if (nums[mid] >= nums[low]) {        low = mid;      } else {        high = mid;      }    }    return result;  }  public static void main(String[] args) {    // 典型輸入,單調升序的數組的一個旋轉    int[] array1 = {3, 4, 5, 1, 2};    System.out.println(getTheMin(array1));    // 有重復數字,并且重復的數字剛好的最小的數字    int[] array2 = {3, 4, 5, 1, 1, 2};    System.out.println(getTheMin(array2));    // 有重復數字,但重復的數字不是第一個數字和最后一個數字    int[] array3 = {3, 4, 5, 1, 2, 2};    System.out.println(getTheMin(array3));    // 有重復的數字,并且重復的數字剛好是第一個數字和最后一個數字    int[] array4 = {1, 0, 1, 1, 1};    System.out.println(getTheMin(array4));    // 單調升序數組,旋轉0個元素,也就是單調升序數組本身    int[] array5 = {1, 2, 3, 4, 5};    System.out.println(getTheMin(array5));    // 數組中只有一個數字    int[] array6 = {2};    System.out.println(getTheMin(array6));    // 數組中數字都相同    int[] array7 = {1, 1, 1, 1, 1, 1, 1};    System.out.println(getTheMin(array7));    // 特殊的不知道如何移動    int[] array8 = {1, 0, 1, 1, 1};    System.out.println(getTheMin(array8));  }}

前面我們提到在旋轉數組中,由于是把遞增排序數組的前面的若干個數字搬到數組后面,因為第一個數字總是大于或者等于最后一個數字,而還有一種特殊情況是移動了 0 個元素,即數組本身,也是它自己的旋轉數組。這種情況本身數組就是有序的了,所以我們只需要返回第一個元素就好了,這也是為什么我先給 result 賦值為 nums[0] 的原因。

上述代碼就完美了嗎?我們通過測試用例并沒有達到我們的要求,我們具體看看 array8 這個輸入。先模擬計算機運行分析一下:

  1. low = 0, high = 4, mid = 2, nums[low] = 1, nums[high] = 1,nums[mid] = 1;
  2. 由于 nums[mid] >= nums[low],故認定 nums[mid] = 1 在左邊遞增子數組中;
  3. 所以更新 high = mid = 2,mid = (low+high)/2 = 1;
  4. nums[low] = 1,nums[mid] = 1,nums[high] = 1;
  5. high - low ≠ 1,繼續循環;
  6. 由于 nums[mid] >= nums[low],故認定 nums[mid] = 1 在左邊遞增子數組中;
  7. 所以更新 high = mid = 1,由于 high - low = 1,故退出循環,得到 result = 1;

但我們一眼了然,明顯我們的最小值不是 1 ,而是 0 ,所以當 array[low]、array[mid]、array[high] 相等的時候,我們的程序并不知道應該如何移動,按照目前的移動方式就默認 array[mid] 在左邊遞增子數組了,這顯然是不負責任的做法。

我們修正一下代碼:

public class Test08 {  public static int getTheMin(int nums[]) {    if (nums == null || nums.length == 0) {      throw new RuntimeException("input error!");    }    // 如果只有一個元素,直接返回    if (nums.length == 1)      return nums[0];    int result = nums[0];    int low = 0, high = nums.length - 1;    int mid = low;    // 確保 low 下標對應的值在左邊的遞增子數組,high 對應的值在右邊遞增子數組    while (nums[low] >= nums[high]) {      // 確保循環結束條件      if (high - low == 1) {        return nums[high];      }      // 取中間位置      mid = (low + high) / 2;      // 三值相等的特殊情況,則需要從頭到尾查找最小的值      if (nums[mid] == nums[low] && nums[mid] == nums[high]) {        return midInorder(nums, low, high);      }      // 代表中間元素在左邊遞增子數組      if (nums[mid] >= nums[low]) {        low = mid;      } else {        high = mid;      }    }    return result;  }  /**   * 查找數組中的最小值   *   * @param nums 數組   * @param start 數組開始位置   * @param end  數組結束位置   * @return 找到的最小的數字   */  public static int midInorder(int[] nums, int start, int end) {    int result = nums[start];    for (int i = start + 1; i <= end; i++) {      if (result > nums[i])        result = nums[i];    }    return result;  }  public static void main(String[] args) {    // 典型輸入,單調升序的數組的一個旋轉    int[] array1 = {3, 4, 5, 1, 2};    System.out.println(getTheMin(array1));    // 有重復數字,并且重復的數字剛好的最小的數字    int[] array2 = {3, 4, 5, 1, 1, 2};    System.out.println(getTheMin(array2));    // 有重復數字,但重復的數字不是第一個數字和最后一個數字    int[] array3 = {3, 4, 5, 1, 2, 2};    System.out.println(getTheMin(array3));    // 有重復的數字,并且重復的數字剛好是第一個數字和最后一個數字    int[] array4 = {1, 0, 1, 1, 1};    System.out.println(getTheMin(array4));    // 單調升序數組,旋轉0個元素,也就是單調升序數組本身    int[] array5 = {1, 2, 3, 4, 5};    System.out.println(getTheMin(array5));    // 數組中只有一個數字    int[] array6 = {2};    System.out.println(getTheMin(array6));    // 數組中數字都相同    int[] array7 = {1, 1, 1, 1, 1, 1, 1};    System.out.println(getTheMin(array7));    // 特殊的不知道如何移動    int[] array8 = {1, 0, 1, 1, 1};    System.out.println(getTheMin(array8));  }}

我們再用完善的測試用例放進去,測試通過。

總結

本題其實考察的點挺多的,實際上就是考察對二分查找的靈活運用,不少小伙伴死記硬背二分查找必須遵從有序,而沒有學會這個二分查找的思想,這樣會導致只能想到循環查找最小值了。

不少小伙伴在面試中表態,Android 原生態基本都封裝了常用算法,對面試這些無作用的算法表示抗議,其實這是相當愚蠢的。我們不求死記硬背算法的實現,但求學習到其中巧妙的思想。只有不斷地提升自己的思維能力,才能助自己收獲更好的職業發展。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
欧美精品免费在线| 亚洲人成电影网站色xx| 国产亚洲精品一区二555| 国产性色av一区二区| 欧美激情videoshd| 国产日韩欧美在线| 精品亚洲夜色av98在线观看| 久久视频在线视频| 中文字幕精品网| 日本人成精品视频在线| 日韩欧美aⅴ综合网站发布| 91亚洲精品视频| 福利一区福利二区微拍刺激| 午夜精品一区二区三区在线视| 亚洲无av在线中文字幕| 国产精品人成电影| 日韩激情av在线播放| 欧美成人精品影院| 国产欧美日韩中文字幕| 国产亚洲福利一区| 欧美日韩国产成人高清视频| 中文字幕亚洲一区二区三区| 成人啪啪免费看| 亚洲毛片在线观看.| 国产69精品久久久久99| 日韩精品一区二区视频| 久久久在线观看| 亚洲区一区二区| 日韩av在线天堂网| 欧美亚州一区二区三区| 国产精品久久久久久久久久三级| 中文亚洲视频在线| 日韩欧美综合在线视频| 亚洲高清色综合| 久久天天躁夜夜躁狠狠躁2022| 欧美裸体男粗大视频在线观看| 欧美精品国产精品日韩精品| 97视频色精品| 亚洲欧美国产精品久久久久久久| 亚洲一区二区久久| 日韩欧美在线视频日韩欧美在线视频| 欧美午夜激情在线| 欧美另类99xxxxx| 国产一区二区三区三区在线观看| 国产99视频在线观看| 国产一区二区三区在线播放免费观看| 日韩中文有码在线视频| 中文字幕久热精品视频在线| 91精品国产99久久久久久| 亚洲国产欧美精品| 欧美性猛交xxxx乱大交蜜桃| 国产精品自拍视频| 欧美一级高清免费播放| 不卡中文字幕av| 亚洲欧美精品一区| 国内精久久久久久久久久人| 欧美丰满少妇xxxxx| 国产精品久久婷婷六月丁香| 国产99久久精品一区二区 夜夜躁日日躁| 精品成人在线视频| 国产激情视频一区| 久久成年人视频| **欧美日韩vr在线| 欧美丝袜一区二区| 欧美电影免费观看高清完整| 国产精品中文字幕在线| 日韩久久免费电影| 久久99久久99精品中文字幕| 中文字幕国产亚洲| 精品国产乱码久久久久久虫虫漫画| 日韩中文字幕视频在线| 国产成人久久久精品一区| 日韩欧美在线视频日韩欧美在线视频| 中文字幕欧美日韩精品| 青青久久av北条麻妃黑人| 久久福利视频导航| 日韩av在线网址| 日韩av快播网址| 久久久久国色av免费观看性色| 久久精品视频亚洲| 亚洲国产精品小视频| 色综合伊人色综合网| 免费成人高清视频| 成人做爰www免费看视频网站| 国产精品视频一区二区高潮| 日韩久久精品电影| 91亚洲国产成人久久精品网站| 亚洲成人免费网站| 日韩欧美第一页| 91亚洲精品久久久| 亚洲午夜激情免费视频| 亚洲国产精品一区二区久| 日韩欧美在线观看视频| 欧美精品福利在线| 久久久精品2019中文字幕神马| 久久6精品影院| 久久久之久亚州精品露出| 国产91热爆ts人妖在线| 欧美日韩精品国产| 国产午夜精品视频免费不卡69堂| 亚洲在线免费看| 国产成人激情小视频| 日本精品久久久| 国产成人黄色av| www.日本久久久久com.| 日本不卡视频在线播放| 国产视频久久久| 久久久精品视频在线观看| 欧美怡春院一区二区三区| 成人午夜在线视频一区| 日本视频久久久| 国产一区二区三区中文| 91精品啪在线观看麻豆免费| 国产精品视频在线播放| 亚洲综合小说区| 成人激情视频网| 性欧美xxxx视频在线观看| zzijzzij亚洲日本成熟少妇| 国产精品一区二区三区在线播放| 麻豆一区二区在线观看| 成人欧美一区二区三区黑人| 成人免费淫片视频软件| 欧美又大又硬又粗bbbbb| 国产精品电影网| 欧美福利小视频| 97久久超碰福利国产精品…| 日韩在线观看网站| 欧美视频免费在线| 91久久久久久久久| 亚洲国产欧美精品| 国产精品一区二区三区久久久| 欧美激情精品久久久久久变态| 成人啪啪免费看| 国产精品嫩草影院久久久| 欧美激情中文网| 亚洲精品电影在线| 国产69精品99久久久久久宅男| 日韩经典第一页| 一区二区成人av| 777国产偷窥盗摄精品视频| 国产丝袜精品视频| 懂色av中文一区二区三区天美| 欧美另类高清videos| 亚洲乱码国产乱码精品精天堂| 亚洲一区二区福利| 日韩在线免费观看视频| 日韩欧美精品在线观看| 亚洲欧洲激情在线| 日韩欧美中文免费| 日韩美女av在线免费观看| 欧美成人午夜免费视在线看片| 欧美俄罗斯性视频| 亚洲精品99久久久久中文字幕| 欧美成人一区在线| 亚洲女在线观看| 亚洲高清色综合| 欧美劲爆第一页| 国产精品麻豆va在线播放| 日本久久精品视频| 久久国产精彩视频| 久久亚洲国产成人| 日韩美女免费线视频| 国产精品美女主播在线观看纯欲| 91久久精品国产91性色|