欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

面試題:Java 實現(xiàn)查找旋轉(zhuǎn)數(shù)組的最小數(shù)字

 更新時間:2018年07月04日 10:24:48   作者:nanchen2251  
這篇文章主要介紹了Java 實現(xiàn)查找旋轉(zhuǎn)數(shù)組的最小數(shù)字,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧

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

面試題:打印出旋轉(zhuǎn)數(shù)組的最小數(shù)字

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

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

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) {
    // 典型輸入,單調(diào)升序的數(shù)組的一個旋轉(zhuǎn)
    int[] array1 = {3, 4, 5, 1, 2};
    System.out.println(getTheMin(array1));

    // 有重復數(shù)字,并且重復的數(shù)字剛好的最小的數(shù)字
    int[] array2 = {3, 4, 5, 1, 1, 2};
    System.out.println(getTheMin(array2));

    // 有重復數(shù)字,但重復的數(shù)字不是第一個數(shù)字和最后一個數(shù)字
    int[] array3 = {3, 4, 5, 1, 2, 2};
    System.out.println(getTheMin(array3));

    // 有重復的數(shù)字,并且重復的數(shù)字剛好是第一個數(shù)字和最后一個數(shù)字
    int[] array4 = {1, 0, 1, 1, 1};
    System.out.println(getTheMin(array4));

    // 單調(diào)升序數(shù)組,旋轉(zhuǎn)0個元素,也就是單調(diào)升序數(shù)組本身
    int[] array5 = {1, 2, 3, 4, 5};
    System.out.println(getTheMin(array5));

    // 數(shù)組中只有一個數(shù)字
    int[] array6 = {2};
    System.out.println(getTheMin(array6));

    // 數(shù)組中數(shù)字都相同
    int[] array7 = {1, 1, 1, 1, 1, 1, 1};
    System.out.println(getTheMin(array7));
  }
}

打印結(jié)果沒什么毛病。不過這樣的方法顯然不是最優(yōu)的,我們看看有沒有辦法找出更加優(yōu)質(zhì)的方法處理。

有序,還要查找?

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

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

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

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

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

  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] 應該是在左邊的遞增子數(shù)組中。
  4. 更新 low = mid = 2,num[low] = 5,mid = (low+high)/2 = 3,num[mid] = 1;
  5. high - low ≠ 1 ,繼續(xù)更新
  6. 由于 num[mid] < num[high],所以斷定 num[mid] = 1 位于右邊的自增子數(shù)組中;
  7. 更新 high = mid = 3,由于 high - mid = 1,所以結(jié)束循環(huán),得到最小值 num[high] = 1;

我們再來看看 Java 中如何用代碼實現(xiàn)這個思路:

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 下標對應的值在左邊的遞增子數(shù)組,high 對應的值在右邊遞增子數(shù)組
    while (nums[low] >= nums[high]) {
      // 確保循環(huán)結(jié)束條件
      if (high - low == 1) {
        return nums[high];
      }
      // 取中間位置
      mid = (low + high) / 2;
      // 代表中間元素在左邊遞增子數(shù)組
      if (nums[mid] >= nums[low]) {
        low = mid;
      } else {
        high = mid;
      }
    }
    return result;
  }

  public static void main(String[] args) {
    // 典型輸入,單調(diào)升序的數(shù)組的一個旋轉(zhuǎn)
    int[] array1 = {3, 4, 5, 1, 2};
    System.out.println(getTheMin(array1));

    // 有重復數(shù)字,并且重復的數(shù)字剛好的最小的數(shù)字
    int[] array2 = {3, 4, 5, 1, 1, 2};
    System.out.println(getTheMin(array2));

    // 有重復數(shù)字,但重復的數(shù)字不是第一個數(shù)字和最后一個數(shù)字
    int[] array3 = {3, 4, 5, 1, 2, 2};
    System.out.println(getTheMin(array3));

    // 有重復的數(shù)字,并且重復的數(shù)字剛好是第一個數(shù)字和最后一個數(shù)字
    int[] array4 = {1, 0, 1, 1, 1};
    System.out.println(getTheMin(array4));

    // 單調(diào)升序數(shù)組,旋轉(zhuǎn)0個元素,也就是單調(diào)升序數(shù)組本身
    int[] array5 = {1, 2, 3, 4, 5};
    System.out.println(getTheMin(array5));

    // 數(shù)組中只有一個數(shù)字
    int[] array6 = {2};
    System.out.println(getTheMin(array6));

    // 數(shù)組中數(shù)字都相同
    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));
  }
}

前面我們提到在旋轉(zhuǎn)數(shù)組中,由于是把遞增排序數(shù)組的前面的若干個數(shù)字搬到數(shù)組后面,因為第一個數(shù)字總是大于或者等于最后一個數(shù)字,而還有一種特殊情況是移動了 0 個元素,即數(shù)組本身,也是它自己的旋轉(zhuǎn)數(shù)組。這種情況本身數(shù)組就是有序的了,所以我們只需要返回第一個元素就好了,這也是為什么我先給 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 在左邊遞增子數(shù)組中;
  3. 所以更新 high = mid = 2,mid = (low+high)/2 = 1;
  4. nums[low] = 1,nums[mid] = 1,nums[high] = 1;
  5. high - low ≠ 1,繼續(xù)循環(huán);
  6. 由于 nums[mid] >= nums[low],故認定 nums[mid] = 1 在左邊遞增子數(shù)組中;
  7. 所以更新 high = mid = 1,由于 high - low = 1,故退出循環(huán),得到 result = 1;

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

我們修正一下代碼:

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 下標對應的值在左邊的遞增子數(shù)組,high 對應的值在右邊遞增子數(shù)組
    while (nums[low] >= nums[high]) {
      // 確保循環(huán)結(jié)束條件
      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);
      }
      // 代表中間元素在左邊遞增子數(shù)組
      if (nums[mid] >= nums[low]) {
        low = mid;
      } else {
        high = mid;
      }
    }
    return result;
  }

  /**
   * 查找數(shù)組中的最小值
   *
   * @param nums 數(shù)組
   * @param start 數(shù)組開始位置
   * @param end  數(shù)組結(jié)束位置
   * @return 找到的最小的數(shù)字
   */
  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) {
    // 典型輸入,單調(diào)升序的數(shù)組的一個旋轉(zhuǎn)
    int[] array1 = {3, 4, 5, 1, 2};
    System.out.println(getTheMin(array1));

    // 有重復數(shù)字,并且重復的數(shù)字剛好的最小的數(shù)字
    int[] array2 = {3, 4, 5, 1, 1, 2};
    System.out.println(getTheMin(array2));

    // 有重復數(shù)字,但重復的數(shù)字不是第一個數(shù)字和最后一個數(shù)字
    int[] array3 = {3, 4, 5, 1, 2, 2};
    System.out.println(getTheMin(array3));

    // 有重復的數(shù)字,并且重復的數(shù)字剛好是第一個數(shù)字和最后一個數(shù)字
    int[] array4 = {1, 0, 1, 1, 1};
    System.out.println(getTheMin(array4));

    // 單調(diào)升序數(shù)組,旋轉(zhuǎn)0個元素,也就是單調(diào)升序數(shù)組本身
    int[] array5 = {1, 2, 3, 4, 5};
    System.out.println(getTheMin(array5));

    // 數(shù)組中只有一個數(shù)字
    int[] array6 = {2};
    System.out.println(getTheMin(array6));

    // 數(shù)組中數(shù)字都相同
    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));

  }
}

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

總結(jié)

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

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

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。

相關(guān)文章

  • 基于SpringBoot+Redis的Session共享與單點登錄詳解

    基于SpringBoot+Redis的Session共享與單點登錄詳解

    這篇文章主要介紹了基于SpringBoot+Redis的Session共享與單點登錄,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友可以參考下
    2019-07-07
  • 使用SpringCache操作Redis緩存數(shù)據(jù)的示例代碼

    使用SpringCache操作Redis緩存數(shù)據(jù)的示例代碼

    SpringCache是一個框架,實現(xiàn)了基于注解的緩存功能,只需要簡單的加一個注解,就能實現(xiàn)緩存功能,本文給大家介紹了如何使用SpringCache操作Redis緩存數(shù)據(jù),文中有相關(guān)的代碼示例供大家參考,需要的朋友可以參考下
    2024-01-01
  • mybatis中映射文件include標簽的應用

    mybatis中映射文件include標簽的應用

    這篇文章主要介紹了mybatis中映射文件include標簽的應用,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-11-11
  • JPA多條件復雜SQL動態(tài)分頁查詢功能

    JPA多條件復雜SQL動態(tài)分頁查詢功能

    這篇文章主要介紹了JPA多條件復雜SQL動態(tài)分頁查詢功能,本文通過實例代碼給大家介紹的非常詳細,具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-09-09
  • Java判斷時間段內(nèi)文件是否更新的方法

    Java判斷時間段內(nèi)文件是否更新的方法

    這篇文章主要介紹了Java判斷時間段內(nèi)文件是否更新的方法,通過實例形式講述了定時器、類加載器及線程等方法實現(xiàn)判斷文件更新的功能,具有一定的參考借鑒價值,需要的朋友可以參考下
    2014-12-12
  • 一篇文章帶你初步認識Maven

    一篇文章帶你初步認識Maven

    這篇文章主要為大家初步認識了Maven,具有一定的參考價值,感興趣的小伙伴們可以參考一下,希望能夠給你帶來幫助
    2022-01-01
  • MyBatis獲取參數(shù)值的兩種方式詳解

    MyBatis獲取參數(shù)值的兩種方式詳解

    本文主要介紹了MyBatis獲取參數(shù)值的兩種方式詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-03-03
  • Java的遞歸算法詳解

    Java的遞歸算法詳解

    Java遞歸算法是基于Java語言實現(xiàn)的遞歸算法。遞歸算法對解決一大類問題很有效,它可以使算法簡潔和易于理解。接下來通過本文給大家介紹Java遞歸算法相關(guān)知識,感興趣的朋友一起學習吧
    2021-09-09
  • 老生常談java中的數(shù)組初始化

    老生常談java中的數(shù)組初始化

    下面小編就為大家?guī)硪黄仙U刯ava中的數(shù)組初始化。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-04-04
  • java中Map集合的常用方法總結(jié)大全

    java中Map集合的常用方法總結(jié)大全

    開發(fā)中最常用的就是List集合和Map集合,Map集合是基于java核心類java.util中的,下面這篇文章主要給大家總結(jié)介紹了關(guān)于java中Map集合的一些常用方法,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2024-01-01

最新評論