374.Guess Number Higher or Lower(Easy)

We are playing the Guess Game. The game is as follows:
I pick a number from 1 to n. You have to guess which number I picked.
Every time you guess wrong, I'll tell you whether the number is higher or lower.
You call a pre-defined API guess(int num) which returns 3 possible results (-1, 1, or 0):
我們來玩一個(gè)猜數(shù)游戲蒿褂,游戲規(guī)則是這樣的:我從1~n中選擇一個(gè)數(shù),你需要猜我選的數(shù)是哪一個(gè),如果你猜錯(cuò)的話死讹,我就會(huì)告訴你你猜的數(shù)比我選擇的數(shù)大還是小岂却,你可以調(diào)用一個(gè)API guess(int num) 泉唁,返回三個(gè)可能的結(jié)果雷酪,-1表示比我的數(shù)小捂龄,1表示比我的數(shù)大释涛,0表示猜中了。

  這個(gè)問題用二分查找即可倦沧,重點(diǎn)只在于確定查找的范圍

For example

n = 10, I pick 6.
Return 6.

My Solution

(Java) Version 1 Time: 1ms:

  用兩個(gè)變量來表示上界和下界唇撬,每次都猜測(cè)范圍1/2位置的數(shù),然后根據(jù)返回的大小關(guān)系確定新的邊界刀脏,踩過的坑是局荚,如果直接用(up - down) / 2 + down來確定中間的數(shù),那么將永遠(yuǎn)不會(huì)查找到作為邊界的兩個(gè)數(shù)愈污,第二坑是我本來想用0~n+1來把邊界的兩個(gè)數(shù)放進(jìn)范圍里,但是果然測(cè)試?yán)锩嬗衖nt的最大值轮傍,這就意味著n+1之后會(huì)溢出暂雹,所以只好在一開始單獨(dú)對(duì)邊界進(jìn)行判斷。

/* The guess API is defined in the parent class GuessGame.
   @param num, your guess
   @return -1 if my number is lower, 1 if my number is higher, otherwise return 0
      int guess(int num); */

public class Solution extends GuessGame {
    public int guessNumber(int n) {
        if(n == 1)return 1;
        else if(guess(n) == 0)return n;
        int down = 0, up = n;
        int key = (up - down) / 2;
        int flag = guess(key);
        while(flag != 0){
            if(flag == 1){
                down = key;
                key = (up - down) / 2 + down;
                flag = guess(key);
            }else{
                up = key;
                key = (up - down) / 2 + down;
                flag = guess(key);
            }
        }
        return key;
    }
}

(Java) Version 2 Time: 1ms (By LeetCode):

*  來自官方的解答创夜,二分查找杭跪,寫法比我的高明得多,起碼邊界問題在循環(huán)中解決了,我第一時(shí)間沒有想到的是解答中的循環(huán)條件……
  時(shí)間復(fù)雜度為O(log2n)

/* The guess API is defined in the parent class GuessGame.
   @param num, your guess
   @return -1 if my number is lower, 1 if my number is higher, otherwise return 0
      int guess(int num); */

public class Solution extends GuessGame {
    public int guessNumber(int n) {
        int low = 1;
        int high = n;
        while (low <= high) {
            int mid = low + (high - low) / 2;
            int res = guess(mid);
            if (res == 0)
                return mid;
            else if (res < 0)
                high = mid - 1;
            else
                low = mid + 1;
        }
        return -1;
    }
}

(Java) Version 3 Time: 1ms (By LeetCode):

  來自官方的解答涧尿,三分查找
  In Binary Search, we choose the middle element as the pivot in splitting. In Ternary Search, we choose two pivots (say m1 and m2) such that the given range is divided into three equal parts. If the required number numnum is less than m1 then we apply ternary search on the left segment of m1. If numnum lies between m1 and m2, we apply ternary search between m1 and m2. Otherwise we will search in the segment right to m2.
  簡單來說就是我們?cè)诙植檎业臅r(shí)候把數(shù)平均分為兩部分系奉,在三分查找中我們把數(shù)平均分為三部分
  時(shí)間復(fù)雜度為O(log3n)

/* The guess API is defined in the parent class GuessGame.
   @param num, your guess
   @return -1 if my number is lower, 1 if my number is higher, otherwise return 0
      int guess(int num); */

public class Solution extends GuessGame {
    public int guessNumber(int n) {
        int low = 1;
        int high = n;
        while (low <= high) {
            int mid1 = low + (high - low) / 3;
            int mid2 = high - (high - low) / 3;
            int res1 = guess(mid1);
            int res2 = guess(mid2);
            if (res1 == 0)
                return mid1;
            if (res2 == 0)
                return mid2;
            else if (res1 < 0)
                high = mid1 - 1;
            else if (res2 > 0)
                low = mid2 + 1;
            else {
                low = mid1 + 1;
                high = mid2 - 1;
            }
        }
        return -1;
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市姑廉,隨后出現(xiàn)的幾起案子缺亮,更是在濱河造成了極大的恐慌,老刑警劉巖桥言,帶你破解...
    沈念sama閱讀 211,042評(píng)論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件萌踱,死亡現(xiàn)場離奇詭異,居然都是意外死亡号阿,警方通過查閱死者的電腦和手機(jī)并鸵,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 89,996評(píng)論 2 384
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來扔涧,“玉大人园担,你說我怎么就攤上這事】菀梗” “怎么了弯汰?”我有些...
    開封第一講書人閱讀 156,674評(píng)論 0 345
  • 文/不壞的土叔 我叫張陵,是天一觀的道長卤档。 經(jīng)常有香客問我蝙泼,道長,這世上最難降的妖魔是什么劝枣? 我笑而不...
    開封第一講書人閱讀 56,340評(píng)論 1 283
  • 正文 為了忘掉前任汤踏,我火速辦了婚禮,結(jié)果婚禮上舔腾,老公的妹妹穿的比我還像新娘溪胶。我一直安慰自己,他們只是感情好稳诚,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,404評(píng)論 5 384
  • 文/花漫 我一把揭開白布哗脖。 她就那樣靜靜地躺著,像睡著了一般扳还。 火紅的嫁衣襯著肌膚如雪才避。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,749評(píng)論 1 289
  • 那天氨距,我揣著相機(jī)與錄音桑逝,去河邊找鬼。 笑死俏让,一個(gè)胖子當(dāng)著我的面吹牛楞遏,可吹牛的內(nèi)容都是我干的茬暇。 我是一名探鬼主播,決...
    沈念sama閱讀 38,902評(píng)論 3 405
  • 文/蒼蘭香墨 我猛地睜開眼寡喝,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼糙俗!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起预鬓,我...
    開封第一講書人閱讀 37,662評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤巧骚,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后珊皿,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體网缝,經(jīng)...
    沈念sama閱讀 44,110評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,451評(píng)論 2 325
  • 正文 我和宋清朗相戀三年蟋定,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了粉臊。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,577評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡驶兜,死狀恐怖扼仲,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情抄淑,我是刑警寧澤屠凶,帶...
    沈念sama閱讀 34,258評(píng)論 4 328
  • 正文 年R本政府宣布,位于F島的核電站肆资,受9級(jí)特大地震影響矗愧,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜郑原,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,848評(píng)論 3 312
  • 文/蒙蒙 一唉韭、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧犯犁,春花似錦属愤、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,726評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至涣澡,卻和暖如春贱呐,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背入桂。 一陣腳步聲響...
    開封第一講書人閱讀 31,952評(píng)論 1 264
  • 我被黑心中介騙來泰國打工吼句, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人事格。 一個(gè)月前我還...
    沈念sama閱讀 46,271評(píng)論 2 360
  • 正文 我出身青樓惕艳,卻偏偏與公主長得像,于是被迫代替她去往敵國和親驹愚。 傳聞我的和親對(duì)象是個(gè)殘疾皇子远搪,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,452評(píng)論 2 348

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