76. 最長(zhǎng)上升子序列

描述

給定一個(gè)整數(shù)序列,找到最長(zhǎng)上升子序列(LIS),返回LIS的長(zhǎng)度埠通。

說明

最長(zhǎng)上升子序列的定義:
最長(zhǎng)上升子序列問題是在一個(gè)無序的給定序列中找到一個(gè)盡可能長(zhǎng)的由低到高排列的子序列骑祟,這種子序列不一定是連續(xù)的或者唯一的。
https://en.wikipedia.org/wiki/Longest_increasing_subsequence

樣例

給出[5,4,1,2,3]态贤,LIS 是[1,2,3]舱呻,返回3
給出[4,2,4,5,3,7],LIS 是[2,4,5,7]抵卫,返回4

挑戰(zhàn)

要求時(shí)間復(fù)雜度為O(n^2) 或者 O(nlogn)

注意

動(dòng)態(tài)規(guī)劃求的是方案的個(gè)數(shù)而不是具體的方案狮荔,即求出最長(zhǎng)子序列的長(zhǎng)度,但最長(zhǎng)子序列究竟由哪些位置構(gòu)成無法得知

代碼

  1. 動(dòng)態(tài)規(guī)劃O(n ^ 2)

思路
本題屬于坐標(biāo)型動(dòng)態(tài)規(guī)劃介粘,也叫接龍型動(dòng)態(tài)規(guī)劃殖氏,但坐標(biāo)并沒有明顯給出,要想成為當(dāng)前元素的上一個(gè)坐標(biāo)需要滿足兩個(gè)條件:
a. 下標(biāo)位于當(dāng)前元素左邊
b. 數(shù)組中對(duì)應(yīng)的值小于當(dāng)前元素

數(shù)組當(dāng)中每一個(gè)元素都可以成為最長(zhǎng)序列的起點(diǎn)姻采,所以我們初始定義一個(gè) f 數(shù)組雅采,f[i] 記錄數(shù)組中相應(yīng)下標(biāo)對(duì)應(yīng)的最長(zhǎng)序列大小,初始化定義 f[i] = 1慨亲,如果存在下標(biāo) j 滿足上述 a 和 b 兩個(gè)條件且滿足 f[j] + 1 > f[i]婚瓜,即 f[j] 對(duì)應(yīng)的最長(zhǎng)序列接上 nums[j] 組成的新序列長(zhǎng)度大于 f[i],那得到新的狀態(tài)刑棵,需要更新 f[i] 的值巴刻,f 數(shù)組中最大的值即為最長(zhǎng)上升子序列

狀態(tài)轉(zhuǎn)移方程為f[i] = max{1, f[j] + 1} (j = 1, 2, ..., i - 1, 且A[j] < A[i])

public class Solution {
    /**
     * @param nums: The integer array
     * @return: The length of LIS (longest increasing subsequence)
     */
    public int longestIncreasingSubsequence(int[] nums) {
        // f數(shù)組存儲(chǔ)著當(dāng)前index對(duì)應(yīng)的最長(zhǎng)上升子序列
        int []f = new int[nums.length];
        int max = 0;
        // 每個(gè)點(diǎn)都可以成為起點(diǎn),初始化令f[i] = 1
        // 如果只令f[0] = 1蛉签,代表只有index = 0才能做起點(diǎn)
        for (int i = 0; i < nums.length; i++) {
            f[i] = 1;
            // index小于i且值小于nums[i]的數(shù)才能做為上一個(gè)點(diǎn)的候選     
            for (int j = 0; j < i; j++) {
                if (nums[j] < nums[i]) {
                    f[i] = f[i] > f[j] + 1 ? f[i] : f[j] + 1;
                }
            }
            // 從所有候選點(diǎn)里找到最長(zhǎng)的
            if (f[i] > max) {
                max = f[i];
            }
        }
        return max;
    }
}
  1. 二分法 O(log n)

思路
O(n^2)的算法復(fù)雜度高的原因就在于必須在1 ~ i-1中枚舉找到最大的 f[j] 的值才能最終更新f[i]的值胡陪,上一種算法的狀態(tài)轉(zhuǎn)移方程為 f[i] = max{1, f[j] + 1} (j = 1, 2, ..., i - 1, 且A[j] < A[i])沥寥,f[i] 代表當(dāng)前下標(biāo)對(duì)應(yīng)的最長(zhǎng)子序列。
現(xiàn)在柠座,我們仔細(xì)考慮計(jì)算 f[i] 時(shí)的情況邑雅。假設(shè)有兩個(gè)元素nums[x]和nums[y],滿足

  1. x < y < i
  2. nums[x] < nums[y] < nums[i]
  3. f[x] = f[y]

此時(shí)妈经,選擇 f[x] 和選擇 f[y] 都可以得到同樣的 f[i] 值淮野,那么,在最長(zhǎng)上升子序列的這個(gè)位置中吹泡,應(yīng)該選擇nums[x]還是應(yīng)該選擇nums[y]呢骤星?

很明顯,選擇nums[x]比選擇nums[y]要好荞胡。如果在nums[x+1] ... nums[i-1]這一段中妈踊,如果存在nums[z],nums[x] < nums[z] < nums[y]泪漂,則選擇nums[x]相比廊营,可能會(huì)得到更長(zhǎng)的上升子序列,當(dāng)前子序列構(gòu)成更長(zhǎng)子序列的潛力增大了萝勤。

如此露筒,我們根據(jù)數(shù)組 f 的值進(jìn)行分類。對(duì)于數(shù)組 f 的每一個(gè)取值k敌卓,我們只需要保留滿足f[i] = k的所有nums中的最小值慎式。用數(shù)組minLast記錄這個(gè)值,即minLast[k] = min{nums[i]} (f[i] = k) (minLast數(shù)組中元素按升序存儲(chǔ))
我們?cè)诒闅vnums數(shù)組的過程中趟径,將在minLast數(shù)組中找到第一個(gè)大于當(dāng)前nums[i]的元素瘪吏,用nums[i]替換該元素就可以實(shí)現(xiàn)剛剛說的記錄滿足f[i] = k的所有nums元素中的最小值 (nums中元素是順序遍歷,minLast元素剛剛大于nums[i]說明它們?cè)趎ums之間的所有元素都大于nums[i]蜗巧,也就意味著從minLast中找出的當(dāng)前元素本身到nums[i]之間所有元素不能成為接龍中nums[i]的上一個(gè)元素掌眠,即minLast找出的當(dāng)前元素對(duì)應(yīng)的f數(shù)組值和f[i]是相等的)

因?yàn)槲覀儗?duì)minLast的替換操作,遍歷到數(shù)組最后minLast的長(zhǎng)度也就成為了最長(zhǎng)子序列的長(zhǎng)度幕屹,要說明的是由于動(dòng)態(tài)規(guī)劃的特性我們得到的minLast并不是真正的接龍方案蓝丙,因?yàn)閷?shí)際接龍?zhí)暨x元素時(shí)nums數(shù)組元素的順序不能改變,而minLast元素的順序是插入得到的望拖,所以minLast中的元素并不是實(shí)際方案

public class Solution {
    /**
     * @param nums: The integer array
     * @return: The length of LIS (longest increasing subsequence)
     */
    public int longestIncreasingSubsequence(int[] nums) {
        if (nums == null || nums.length == 0) {
            return 0;
        }        

        // 二分法尋找的是end位置渺尘,所以minLast數(shù)組至少要有兩個(gè)元素才能進(jìn)行插入更新值
        // 所以minLast數(shù)組要多加一個(gè)元素
        // 實(shí)際上minLast[0]會(huì)一直保持Integer.MIN_VALUE不變
        int[] minLast = new int[nums.length + 1];
        minLast[0] = Integer.MIN_VALUE;
        for (int i = 1; i <= nums.length; i++) {
            minLast[i] = Integer.MAX_VALUE;
        }
        
        for (int i = 0; i < nums.length; i++) {
            // find the first number in minLast >= nums[i]
            int index = binarySearch(minLast, nums[i]);
            minLast[index] = nums[i];
        }
        
        for (int i = nums.length; i >= 1; i--) {
            if (minLast[i] != Integer.MAX_VALUE) {
                return i;
            }
        }
        
        return 0;
    }
    
    // find the first number > num
    private int binarySearch(int[] minLast, int num) {
        int start = 0, end = minLast.length - 1;
        while (start + 1 < end) {
            int mid = (end - start) / 2 + start;
            if (minLast[mid] < num) {
                start = mid;
            } else {
                end = mid;
            }
        }
        
        return end;
    }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市说敏,隨后出現(xiàn)的幾起案子鸥跟,更是在濱河造成了極大的恐慌,老刑警劉巖盔沫,帶你破解...
    沈念sama閱讀 216,372評(píng)論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件锌雀,死亡現(xiàn)場(chǎng)離奇詭異蚂夕,居然都是意外死亡迅诬,警方通過查閱死者的電腦和手機(jī)腋逆,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,368評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來侈贷,“玉大人惩歉,你說我怎么就攤上這事∏温” “怎么了撑蚌?”我有些...
    開封第一講書人閱讀 162,415評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)搏屑。 經(jīng)常有香客問我争涌,道長(zhǎng),這世上最難降的妖魔是什么辣恋? 我笑而不...
    開封第一講書人閱讀 58,157評(píng)論 1 292
  • 正文 為了忘掉前任亮垫,我火速辦了婚禮,結(jié)果婚禮上伟骨,老公的妹妹穿的比我還像新娘饮潦。我一直安慰自己,他們只是感情好携狭,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,171評(píng)論 6 388
  • 文/花漫 我一把揭開白布继蜡。 她就那樣靜靜地躺著,像睡著了一般逛腿。 火紅的嫁衣襯著肌膚如雪稀并。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,125評(píng)論 1 297
  • 那天单默,我揣著相機(jī)與錄音碘举,去河邊找鬼。 笑死雕凹,一個(gè)胖子當(dāng)著我的面吹牛殴俱,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播枚抵,決...
    沈念sama閱讀 40,028評(píng)論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼线欲,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了汽摹?” 一聲冷哼從身側(cè)響起李丰,我...
    開封第一講書人閱讀 38,887評(píng)論 0 274
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎逼泣,沒想到半個(gè)月后趴泌,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體舟舒,經(jīng)...
    沈念sama閱讀 45,310評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,533評(píng)論 2 332
  • 正文 我和宋清朗相戀三年嗜憔,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了秃励。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,690評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡吉捶,死狀恐怖夺鲜,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情呐舔,我是刑警寧澤币励,帶...
    沈念sama閱讀 35,411評(píng)論 5 343
  • 正文 年R本政府宣布,位于F島的核電站珊拼,受9級(jí)特大地震影響食呻,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜澎现,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,004評(píng)論 3 325
  • 文/蒙蒙 一仅胞、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧昔头,春花似錦饼问、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,659評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至讹开,卻和暖如春盅视,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背旦万。 一陣腳步聲響...
    開封第一講書人閱讀 32,812評(píng)論 1 268
  • 我被黑心中介騙來泰國(guó)打工闹击, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人成艘。 一個(gè)月前我還...
    沈念sama閱讀 47,693評(píng)論 2 368
  • 正文 我出身青樓赏半,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親淆两。 傳聞我的和親對(duì)象是個(gè)殘疾皇子断箫,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,577評(píng)論 2 353

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