面試 3:查找旋轉(zhuǎn)數(shù)組的最小數(shù)字

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

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

題目:把一個(gè)數(shù)組最開始的若干個(gè)元素搬到數(shù)組的末尾,我們稱之為數(shù)組的旋轉(zhuǎn)玛迄。輸入一個(gè)遞增排序的數(shù)組的一個(gè)旋轉(zhuǎn)由境,輸出旋轉(zhuǎn)數(shù)組的最小元素。例如數(shù)組 {3,4虏杰,5讥蟆,1,2} 為數(shù)組 {1纺阔,2瘸彤,3,4笛钝,5} 的一個(gè)旋轉(zhuǎn)质况,該數(shù)組的最小值為 1。

要想實(shí)現(xiàn)這個(gè)需求很簡單玻靡,我們只需要遍歷一遍數(shù)組结榄,找到最小的值后直接退出循環(huán)。代碼實(shí)現(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ù)組的一個(gè)旋轉(zhuǎn)
        int[] array1 = {3, 4, 5, 1, 2};
        System.out.println(getTheMin(array1));

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

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

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

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

        // 數(shù)組中只有一個(gè)數(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ì)的方法處理。

有序醋拧,還要查找慷嗜?

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

我們可以設(shè)定兩個(gè)下標(biāo) low 和 high,并設(shè)定 mid = (low + high)/2琉用,我們自然就可以找到數(shù)組中間的元素 array[mid]堕绩,如果中間的元素位于前面的遞增數(shù)組,那么它應(yīng)該大于或者等于 low 下標(biāo)對應(yīng)的元素邑时,此時(shí)數(shù)組中最小的元素應(yīng)該位于該元素的后面奴紧,我們可以把 low 下標(biāo)指向該中間元素,這樣可以縮小查找的范圍晶丘。

同樣黍氮,如果中間元素位于后面的遞增子數(shù)組,那么它應(yīng)該小于或者等于 high 下標(biāo)對應(yīng)的元素。此時(shí)該數(shù)組中最小的元素應(yīng)該位于該中間元素的前面沫浆。我們就可以把 high 下標(biāo)更新到中位數(shù)的下標(biāo)觉壶,這樣也可以縮小查找的范圍,移動(dòng)之后的 high 下標(biāo)對應(yīng)的元素仍然在后面的遞增子數(shù)組中件缸。

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

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

  1. input:{3凿叠,4,5嚼吞,1盒件,2}
  2. 此時(shí) low = 0,high = 4舱禽,mid = 2炒刁,對應(yīng)的值分別是:num[low] = 3,num[high] = 2誊稚,num[mid] = 5
  3. 由于 num[mid] > num[low]翔始,所以 num[mid] 應(yīng)該是在左邊的遞增子數(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 中如何用代碼實(shí)現(xiàn)這個(gè)思路:

public class Test08 {

    public static int getTheMin(int nums[]) {
        if (nums == null || nums.length == 0) {
            throw new RuntimeException("input error!");
        }
        // 如果只有一個(gè)元素,直接返回
        if (nums.length == 1)
            return nums[0];
        int result = nums[0];
        int low = 0, high = nums.length - 1;
        int mid;
        // 確保 low 下標(biāo)對應(yīng)的值在左邊的遞增子數(shù)組漫萄,high 對應(yīng)的值在右邊遞增子數(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ù)組的一個(gè)旋轉(zhuǎn)
        int[] array1 = {3, 4, 5, 1, 2};
        System.out.println(getTheMin(array1));

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

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

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

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

        // 數(shù)組中只有一個(gè)數(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));

        // 特殊的不知道如何移動(dòng)
        int[] array8 = {1, 0, 1, 1, 1};
        System.out.println(getTheMin(array8));
    }
}

前面我們提到在旋轉(zhuǎn)數(shù)組中未巫,由于是把遞增排序數(shù)組的前面的若干個(gè)數(shù)字搬到數(shù)組后面窿撬,因?yàn)榈谝粋€(gè)數(shù)字總是大于或者等于最后一個(gè)數(shù)字,而還有一種特殊情況是移動(dòng)了 0 個(gè)元素叙凡,即數(shù)組本身劈伴,也是它自己的旋轉(zhuǎn)數(shù)組。這種情況本身數(shù)組就是有序的了握爷,所以我們只需要返回第一個(gè)元素就好了跛璧,這也是為什么我先給 result 賦值為 nums[0] 的原因。

上述代碼就完美了嗎新啼?我們通過測試用例并沒有達(dá)到我們的要求追城,我們具體看看 array8 這個(gè)輸入。先模擬計(jì)算機(jī)運(yùn)行分析一下:

  1. low = 0, high = 4, mid = 2, nums[low] = 1, nums[high] = 1,nums[mid] = 1;
  2. 由于 nums[mid] >= nums[low]燥撞,故認(rèn)定 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]色洞,故認(rèn)定 nums[mid] = 1 在左邊遞增子數(shù)組中;
  7. 所以更新 high = mid = 1,由于 high - low = 1,故退出循環(huán)茶鉴,得到 result = 1锋玲;

但我們一眼了然,明顯我們的最小值不是 1 涵叮,而是 0 惭蹂,所以當(dāng) array[low]、array[mid]割粮、array[high] 相等的時(shí)候盾碗,我們的程序并不知道應(yīng)該如何移動(dòng),按照目前的移動(dòng)方式就默認(rèn) array[mid] 在左邊遞增子數(shù)組了舀瓢,這顯然是不負(fù)責(zé)任的做法廷雅。

我們修正一下代碼:

public class Test08 {

    public static int getTheMin(int nums[]) {
        if (nums == null || nums.length == 0) {
            throw new RuntimeException("input error!");
        }
        // 如果只有一個(gè)元素,直接返回
        if (nums.length == 1)
            return nums[0];
        int result = nums[0];
        int low = 0, high = nums.length - 1;
        int mid = low;
        // 確保 low 下標(biāo)對應(yīng)的值在左邊的遞增子數(shù)組京髓,high 對應(yīng)的值在右邊遞增子數(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ù)組的一個(gè)旋轉(zhuǎn)
        int[] array1 = {3, 4, 5, 1, 2};
        System.out.println(getTheMin(array1));

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

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

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

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

        // 數(shù)組中只有一個(gè)數(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));

        // 特殊的不知道如何移動(dòng)
        int[] array8 = {1, 0, 1, 1, 1};
        System.out.println(getTheMin(array8));

    }
}

我們再用完善的測試用例放進(jìn)去赶袄,測試通過。

總結(jié)

本題其實(shí)考察的點(diǎn)挺多的抠藕,實(shí)際上就是考察對二分查找的靈活運(yùn)用饿肺,不少小伙伴死記硬背二分查找必須遵從有序,而沒有學(xué)會(huì)這個(gè)二分查找的思想盾似,這樣會(huì)導(dǎo)致只能想到循環(huán)查找最小值了敬辣。

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

這也大概是大家一直到處叫大佬,埋怨自己工資總是跟不上別人的一方面原因吧髓梅。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末拟蜻,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子枯饿,更是在濱河造成了極大的恐慌酝锅,老刑警劉巖,帶你破解...
    沈念sama閱讀 219,110評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件奢方,死亡現(xiàn)場離奇詭異搔扁,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī)蟋字,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,443評論 3 395
  • 文/潘曉璐 我一進(jìn)店門稿蹲,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人鹊奖,你說我怎么就攤上這事。” “怎么了民褂?”我有些...
    開封第一講書人閱讀 165,474評論 0 356
  • 文/不壞的土叔 我叫張陵柳弄,是天一觀的道長。 經(jīng)常有香客問我两蟀,道長网梢,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,881評論 1 295
  • 正文 為了忘掉前任垫竞,我火速辦了婚禮澎粟,結(jié)果婚禮上蛀序,老公的妹妹穿的比我還像新娘。我一直安慰自己活烙,他們只是感情好徐裸,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,902評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著啸盏,像睡著了一般重贺。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上回懦,一...
    開封第一講書人閱讀 51,698評論 1 305
  • 那天气笙,我揣著相機(jī)與錄音,去河邊找鬼怯晕。 笑死潜圃,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的舟茶。 我是一名探鬼主播谭期,決...
    沈念sama閱讀 40,418評論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼吧凉!你這毒婦竟也來了隧出?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,332評論 0 276
  • 序言:老撾萬榮一對情侶失蹤阀捅,失蹤者是張志新(化名)和其女友劉穎胀瞪,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體饲鄙,經(jīng)...
    沈念sama閱讀 45,796評論 1 316
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡凄诞,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,968評論 3 337
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了忍级。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片幔摸。...
    茶點(diǎn)故事閱讀 40,110評論 1 351
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖颤练,靈堂內(nèi)的尸體忽然破棺而出既忆,到底是詐尸還是另有隱情,我是刑警寧澤嗦玖,帶...
    沈念sama閱讀 35,792評論 5 346
  • 正文 年R本政府宣布患雇,位于F島的核電站,受9級特大地震影響宇挫,放射性物質(zhì)發(fā)生泄漏苛吱。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,455評論 3 331
  • 文/蒙蒙 一器瘪、第九天 我趴在偏房一處隱蔽的房頂上張望翠储。 院中可真熱鬧绘雁,春花似錦、人聲如沸援所。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,003評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽住拭。三九已至挪略,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間滔岳,已是汗流浹背杠娱。 一陣腳步聲響...
    開封第一講書人閱讀 33,130評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留谱煤,地道東北人摊求。 一個(gè)月前我還...
    沈念sama閱讀 48,348評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像刘离,于是被迫代替她去往敵國和親睹簇。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,047評論 2 355

推薦閱讀更多精彩內(nèi)容