丑數----java實現

題目:丑數

  • 我們把只包含因子2,3,5的數稱為丑數(Ugly Number).
  • 求按從小到大的順序的第1500個丑數。
  • 例如6,8都是丑數,但14不是,因為它含有因子7.習慣上我們把1當作第一個丑數
方法一:逐個判斷每個整數是不是丑數,直觀但不夠高效:

所謂一個數m是另一個數n的因子,是指n能被m整除,也就是說n%m==0.根據丑數的定義,丑數只能被2,3,5整除。也就是說如果一個數能被2整除,我們把它連續除以2;如果能被3整除,就連續除以3;如果能被5整除,就除以5.如果最后我們得到的是1,那么這個數就是丑數,否則不是。

public boolean isUgly(int number){  
        while(number % 2 == 0)  
            number/=2;  
        while(number % 3 == 0)  
            number /=3;  
        while(number % 5 == 0)  
            number /=5;  
        return (number ==1)? true:false;  
    }

接下來,只需要按照順序判斷每個整數是不是丑數即可。

public int getUglyNumber(int index){  
        if(index <= 0)  
            return 0;  
        int number = 0;  
        int uglyFound = 0;  
        while(uglyFound < index){  
            number++;  
            if(isUgly(number)){  
                ++uglyFound;  
            }  
        }  
        return number;  
    }  

我們只需要在函數getUglyNumber 中傳入參數1500,就能得到第1500個丑數。該算法非常直觀,代碼也非常簡潔,但最大的問題是每個整數都需要計算。即使一個數字不是丑數,我們還是需要對它做求余和除法操作。因此該算法的時間效率不是很高,面試官也不會就此滿足,還會提示我們有更高效的算法。

方法二:創建數組保存已經找到的丑數,用空間換時間的解法:

前面的算法之所以效率低,很大程度上是因為不管一個數是不是丑數我們對它都要作計算。接下來我們試著找到一種只要計算丑數的方法,而不在非丑數的整數上花費時間。根據丑數的定義,丑數應該是另一個丑數乘以2,3,5的結果。因此我們可以創建一個數組,里面的數字是排序好的丑數,每一個丑數都是前面的丑數乘以2,3,5得到的。

這種思路的關鍵在于怎樣確定數組里面的丑數是排序好的。假設數組中已經有若干個丑數排好后存放在數組中,并且把已有的最大的丑數記作M,我們接下來分析如何生成下一個丑數。該丑數肯定是前面某個丑數乘以2,3,5的結果。所以我們首先考慮把已有的每個丑數乘以2.在乘以2的時候,能得到若干個小于或等于M的結果。由于是按照順序生成的,小于或者等于M肯定已經在數組中了,我們不需要再次考慮;還會得到若干個大于M的結果,但我們只需要第一個大于M的結果,因為我們希望丑數是指按從小到大的順序生成的,其他更大的結果以后再說。我們把得到的第一個乘以2后大于M的結果即為M2.同樣,我們把已有的每一個丑數乘以3,5,能得到第一個大于M的結果M3和M5.那么下一個丑數應該是M2,M3,M5。這3個數的最小者。

前面分析的時候,提到把已有的每個丑數分別都乘以2,3,5.事實上這不是必須的,因為已有的丑數都是按順序存放在數組中的。對乘以2而言,肯定存在某一個丑數T2,排在它之前的每一個丑數乘以2得到的結果都會小于已有的最大丑數,在它之后的每一個丑數乘以2得到的結果都會太大。我們只需記下這個丑數的位置,同時每次生成新的丑數的時候,去更新這個T2.對乘以3和5而言,也存在這同樣的T3和T5.
因此java代碼實現如下:

public int getUglyNumber_Solution2(int index){
        if(index <=0)
            return 0;
        int[] uglyArray = new int[index];
        uglyArray[0] = 1;
        int multiply2 = 0;
        int multiply3 = 0;
        int multiply5 = 0;
        for(int i = 1;i<index;i++){
            int min = min(uglyArray[multiply2]*2,uglyArray[multiply3]*3,uglyArray[multiply5]*5);
            uglyArray[i] = min;
            //生成新的丑數后及時更新三個游標
            while(uglyArray[multiply2]*2 <= uglyArray[i])
                ++multiply2;
            while(uglyArray[multiply3]*3 <= uglyArray[i])
                ++multiply3;
            while(uglyArray[multiply5]*5 <= uglyArray[i])
                ++multiply5;
        }
        return uglyArray[index-1];
    }
    public int min(int number1,int number2,int number3){
        int min = (number1<number2)?number1:number2;
        return min <number3?min:number3;
    }

和第一種思路相比,第二種思路不需要在非丑數的整數上做任何計算,因此時間效率有明顯上升。但也需要指出,第二種算法由于需要保存已經生成的丑數,因此需要一個數組,從而增加了空間消耗。如果是求第1500個丑數,將創建一個能容納1500個丑數的數組,這個數組占內存6KB。而第一種思路沒有這樣的內存開銷。總的來說,第二種思路相當于用較少的空間換取了時間效率上的提升。

?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 227,797評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,179評論 3 414
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,628評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,642評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,444評論 6 405
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 54,948評論 1 321
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,040評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,185評論 0 287
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,717評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,602評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,794評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,316評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,045評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,418評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,671評論 1 281
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,414評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,750評論 2 370

推薦閱讀更多精彩內容