LeetCode 周賽上分之旅 #40 結合特征壓縮的數(shù)位 DP 問題

雙周賽 111

T1. 統(tǒng)計和小于目標的下標對數(shù)目(Easy)

  • 標簽:模擬谤辜、排序、相向雙指針

T2. 循環(huán)增長使字符串子序列等于另一個字符串(Medium)

  • 標簽:排序、雙指針

T3. 將三個組排序(Medium)

  • 標簽:狀態(tài)機 DP、LIS 問題弯汰、貪心、二分查找

T4. 范圍中美麗整數(shù)的數(shù)目(Hard)

  • 標簽:數(shù)位 DP湖雹、記憶化

T1. 統(tǒng)計和小于目標的下標對數(shù)目(Easy)

https://leetcode.cn/problems/count-pairs-whose-sum-is-less-than-target/

題解一(模擬)

簡單模擬題咏闪。

class Solution {
    fun countPairs(nums: List<Int>, target: Int): Int {
        var ret = 0
        for (i in 0 until nums.size) {
             for (j in i + 1 until nums.size) {
                 if (nums[i] + nums[j] < target) ret ++
             }
        }
        return ret
    }
}

復雜度分析:

  • 時間復雜度:O(n^2)
  • 空間復雜度:O(1)

題解二(排序 + 相向雙指針)

在題解一中存在很多無意義的比較,我們觀察到配對的順序是無關的劝枣,因此可以考慮利用有序性優(yōu)化時間復雜度汤踏。

  • 先對數(shù)組排序;
  • 利用元素的大小關系舔腾,使用雙指針篩選滿足條件的配對數(shù)溪胶。
class Solution {
    fun countPairs(nums: MutableList<Int>, target: Int): Int {
        nums.sort()
        var ret = 0
        var i = 0
        var j = nums.size - 1
        while (i < j) {
            while (i < j && nums[i] + nums[j] >= target) {
                j--
            }
            if (i == j) break
            ret += j - i
            i++
        }
        return ret
    }
}

復雜度分析:

  • 時間復雜度:O(nlgn) 瓶頸在排序,雙指針時間復雜度為 O(n)稳诚;
  • 空間復雜度:O(lgn) 排序遞歸椈┎保空間。

T2. 循環(huán)增長使字符串子序列等于另一個字符串(Medium)

https://leetcode.cn/problems/make-string-a-subsequence-using-cyclic-increments/

題解(雙指針 + 貪心)

首先閱讀題意扳还,問題相當于從 str1 中選擇若干個位置執(zhí)行 +1 操作后讓 str2 成為 str1 的子序列才避。其次容易想到貪心算法,對于 str1[i] 與 str2[j] 來說氨距,如果 str1[i] 能夠在至多操作 1 次的情況下變?yōu)?str2[j]桑逝,那么讓 i 與 j 匹配是最優(yōu)的。

class Solution {
    fun canMakeSubsequence(str1: String, str2: String): Boolean {
        val U = 26
        val n = str1.length
        val m = str2.length
        var j = 0
        for (i in 0 until n) {
            val x = str1[i] - 'a'
            val y = str2[j] - 'a'
            if ((y - x + U) % U <= 1) {
                if (++j == m) break
            }
        }
        return m == j
    }
}

復雜度分析:

  • 時間復雜度:O(n + m) i 指針和 j 指針最多移動 n + m 次俏让;
  • 空間復雜度:O(1) 僅使用常量級別空間楞遏。

T3. 將三個組排序(Medium)

https://leetcode.cn/problems/sorting-three-groups/

題解一(狀態(tài)機 DP)

根據(jù)題意,我們需要構造出非遞減數(shù)組首昔,并使得操作次數(shù)最小寡喝。

觀察測試用例可以發(fā)現(xiàn)逆序數(shù)是問題的關鍵,如示例 1 [2,1,3,2,1] 中存在 2 → 1勒奇,3 → 2预鬓,2 → 1 的逆序對,且結果正好是 3赊颠。然而這個思路是錯誤的格二,我們可以構造特殊測試用例 [3,3,3,1,1,1] 來驗證劈彪。

那應該怎么解呢?我們發(fā)現(xiàn)問題是可以劃分為子問題的蟋定。定義 dp[i][j] 表示到 [i] 為止構造出以 j 為結尾的非遞減數(shù)組的最少操作次數(shù)粉臊,那么 dp[i+1][j] 可以從 dp[i] 的三個子狀態(tài)轉移過來:

  • dp[i][1] 可以轉移到 dp[i+1][1] 和 dp[i+1][2] 和 dp[i+1][3]
  • dp[i][2] 可以轉移到 dp[i+1][2] 和 dp[i+1][3]
  • dp[i][3] 可以轉移到 dp[i+1][3]

最后草添,求出 dp[n][1]驶兜、dp[n][2] 和 dp[n][3] 中的最小值即為問題的解。

class Solution {
    fun minimumOperations(nums: List<Int>): Int {
        val n = nums.size
        val U = 3
        val dp = Array(n + 1) { IntArray(U + 1) }
        for (i in 1 .. n) {
            for (j in 1 .. U) {
                dp[i][j] = dp[i - 1].slice(1..j).min()!!
                if (j != nums[i - 1]) dp[i][j] += 1
            }
        }
        return dp[n].slice(1..U).min()!!
    }
}

另外远寸,dp[i+1] 只與 dp[i] 有關抄淑,我們可以使用滾動數(shù)組優(yōu)化空間復雜度:

class Solution {
    fun minimumOperations(nums: List<Int>): Int {
        val n = nums.size
        val U = 3
        val dp = IntArray(U + 1)
        for (i in 0 until n) {
            for (j in U downTo 1) { // 逆序
                dp[j] = dp.slice(1..j).min()!!
                if (j != nums[i]) dp[j] += 1
            }
        }
        return dp.slice(1..U).min()!!
    }
}

復雜度分析:

  • 時間復雜度:O(C·n) 僅需要線性時間,其中 C = 9驰后;
  • 空間復雜度:O(U) DP 數(shù)組空間肆资,U = 3。

題解二(LIS 問題)

這道題還有第二種思路灶芝,我們可以計算數(shù)組的最長非遞減子序列長度 LIS郑原,再使用原數(shù)組長度 n - 最長非遞減子序列長度 LIS,就可以得出最少操作次數(shù)夜涕。

LIS 問題有兩個寫法:

寫法 1 · 動態(tài)規(guī)劃

class Solution {
    fun minimumOperations(nums: List<Int>): Int {
        val n = nums.size
        // dp[i] 表示以 [i] 為結尾的 LIS
        val dp = IntArray(n) { 1 }
        var len = 1
        for (i in 0 until n) {
            for (j in 0 until i) {
                if (nums[i] >= nums[j]) dp[i] = Math.max(dp[i], dp[j] + 1)
            }
            len = Math.max(len, dp[i])
        }
        return n - len
    }
}

復雜度分析:

  • 時間復雜度:O(n^2) 內存循環(huán)的時間復雜度是 O(n)犯犁;
  • 空間復雜度:O(n) DP 數(shù)組空間。

寫法 2 · 動態(tài)規(guī)劃 + 貪心 + 二分查找

class Solution {
    fun minimumOperations(nums: List<Int>): Int {
        val n = nums.size
        val dp = IntArray(n + 1)
        var len = 1
        dp[len] = nums[0]
        for (i in 1 until n) {
            if (nums[i] >= dp[len]) {
                dp[++len] = nums[i]
            } else {
                // 二分查找維護增長更慢的序列:尋找大于 nums[i] 的第一個數(shù)
                var left = 1
                var right = len
                while (left < right) {
                    val mid = (left + right) ushr 1
                    if (dp[mid] <= nums[i]) {
                        left = mid + 1
                    } else {
                        right = mid 
                    }
                }
                dp[left] = nums[i]
            }
        }
        return n - len
    }
}

復雜度分析:

  • 時間復雜度:O(nlgn) 單次二分查找的時間復雜度是 O(lgn)女器;
  • 空間復雜度:O(n) DP 數(shù)組空間酸役。

相似題目:


T4. 范圍中美麗整數(shù)的數(shù)目(Hard)

https://leetcode.cn/problems/number-of-beautiful-integers-in-the-range/

題解(數(shù)位 DP + 記憶化)

近期經(jīng)常出現(xiàn)數(shù)位 DP 的題目。

  • 1驾胆、數(shù)位 DP: 我們定義 dp[i, pre, diff, isNumber, isLimit] 表示從第 i 位開始的合法方案數(shù)涣澡,其中:
    • pre 表示已經(jīng)選擇的數(shù)位前綴的值,當填入第 i 位的數(shù)字 choice 后更新為 pre * 10 + choice丧诺,在終止條件時判斷 pre % k == 0入桂;
    • diff 表示已選擇的數(shù)位中奇數(shù)和偶數(shù)的差值,奇數(shù) + 1驳阎,而偶數(shù) - 1抗愁,在終止條件時判斷 diff == 0;
    • isNumber 表示已填數(shù)位是否構造出合法數(shù)字搞隐;
    • isLimit 表示當前數(shù)位是否被當前數(shù)位的最大值約束驹愚。
  • 2、差值: 要計算出 [low, high] 之間的合法方案數(shù)劣纲,我們可以計算出 [0, high] 和 [0, low] 之間合法方案數(shù)的差值逢捺;
  • 3、記憶化: 對于相同 dp[i, …] 子問題癞季,可能會重復計算劫瞳,可以使用記憶化優(yōu)化時間復雜度:
  • 4倘潜、特征壓縮: 由于所有的備選數(shù)的 pre 都是不用的,這會導致永遠不會命中備忘錄志于,我們需要找到不同前綴的特征涮因。
  • 5、取模公式: 如果 (pre * 10 + choice) \% k == 0伺绽,那么有 ((pre \% k) * 10 + choice) \% k == 0待锈,我們可以提前對 pre 取模抽取出特征因子伤提。

(a + b) \% mod == ((a \% mod) + (b \% mod)) \% mod
(a · b) \% mod == ((a \% mod) · (b \% mod)) \% mod

class Solution {
    
    private val U = 10
    
    fun numberOfBeautifulIntegers(low: Int, high: Int, k: Int): Int {
        return count(high, k) - count(low - 1, k)
    }
    
    private fun count(num: Int, k: Int): Int {
        // <i, diff, k>
        val memo = Array(U) { Array(U + U) { IntArray(k) { -1 }} }
        return f(memo, "$num", k, 0, 0, 0, true, false)
    }
    
    private fun f(memo: Array<Array<IntArray>>, str: String, k: Int, i: Int, mod: Int, diff: Int, isLimit: Boolean, isNumber: Boolean): Int {
        // 終止條件
        if (i == str.length) return if (0 != diff || mod % k != 0) 0 else 1
        // 讀備忘錄
        if (!isLimit && isNumber && -1 != memo[i][diff + U][mod]) {
            return memo[i][diff + U][mod] // 由于 diff 的取值是 [-10,10],我們增加一個 U 的偏移
        }
        val upper = if (isLimit) str[i] - '0' else 9
        var ret = 0
        for (choice in 0 .. upper) {
            val newMod = (mod * 10 + choice) % k // 特征因子
            if (!isNumber && choice == 0) {
                ret += f(memo, str, k, i + 1, 0, 0, false, false)
                continue
            } 
            if (choice % 2 == 0) {
                ret += f(memo, str, k, i + 1, newMod, diff + 1, isLimit && choice == upper, true)
            } else {
                ret += f(memo, str, k, i + 1, newMod, diff - 1, isLimit && choice == upper, true)
            }   
        }
        // 寫備忘錄
        if (!isLimit && isNumber) {
            memo[i][diff + U][mod] = ret
        }
        return ret
    }
}

復雜度分析:

  • 時間復雜度:O(C^2·k·D) 其中 C 為最大數(shù)位長度 10,D 為選擇方案 10随常。狀態(tài)數(shù)為 “i 的值域 * diff 的值域 * mod 的值域” = C^2·k胞得,單個狀態(tài)的時間復雜度是 O(D)慢洋,整體的時間復雜度是狀態(tài)數(shù) · 狀態(tài)時間 = O(C^2·k·D)垮抗;
  • 空間復雜度:O(C^2·k) 備忘錄空間。

相似題目:


推薦閱讀

LeetCode 上分之旅系列往期回顧:

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末惩妇,一起剝皮案震驚了整個濱河市株汉,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌歌殃,老刑警劉巖乔妈,帶你破解...
    沈念sama閱讀 216,324評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異挺份,居然都是意外死亡褒翰,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評論 3 392
  • 文/潘曉璐 我一進店門匀泊,熙熙樓的掌柜王于貴愁眉苦臉地迎上來优训,“玉大人,你說我怎么就攤上這事各聘〈Х牵” “怎么了?”我有些...
    開封第一講書人閱讀 162,328評論 0 353
  • 文/不壞的土叔 我叫張陵躲因,是天一觀的道長早敬。 經(jīng)常有香客問我,道長大脉,這世上最難降的妖魔是什么搞监? 我笑而不...
    開封第一講書人閱讀 58,147評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮镰矿,結果婚禮上琐驴,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好绝淡,可當我...
    茶點故事閱讀 67,160評論 6 388
  • 文/花漫 我一把揭開白布宙刘。 她就那樣靜靜地躺著,像睡著了一般牢酵。 火紅的嫁衣襯著肌膚如雪悬包。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,115評論 1 296
  • 那天馍乙,我揣著相機與錄音布近,去河邊找鬼。 笑死潘拨,一個胖子當著我的面吹牛吊输,可吹牛的內容都是我干的饶号。 我是一名探鬼主播铁追,決...
    沈念sama閱讀 40,025評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼茫船!你這毒婦竟也來了琅束?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 38,867評論 0 274
  • 序言:老撾萬榮一對情侶失蹤算谈,失蹤者是張志新(化名)和其女友劉穎涩禀,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體然眼,經(jīng)...
    沈念sama閱讀 45,307評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡艾船,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,528評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了高每。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片屿岂。...
    茶點故事閱讀 39,688評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖鲸匿,靈堂內的尸體忽然破棺而出爷怀,到底是詐尸還是另有隱情,我是刑警寧澤带欢,帶...
    沈念sama閱讀 35,409評論 5 343
  • 正文 年R本政府宣布运授,位于F島的核電站,受9級特大地震影響乔煞,放射性物質發(fā)生泄漏吁朦。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,001評論 3 325
  • 文/蒙蒙 一渡贾、第九天 我趴在偏房一處隱蔽的房頂上張望逗宜。 院中可真熱鬧,春花似錦、人聲如沸锦溪。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽刻诊。三九已至防楷,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間则涯,已是汗流浹背复局。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評論 1 268
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留粟判,地道東北人亿昏。 一個月前我還...
    沈念sama閱讀 47,685評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像档礁,于是被迫代替她去往敵國和親角钩。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,573評論 2 353

推薦閱讀更多精彩內容