Given a non-empty string s and a dictionary wordDict containing a list of non-emptywords, determine if s can be segmented into a space-separated sequence of one or more dictionary words.
Note:
- The same word in the dictionary may be reused multiple times in the segmentation.
- You may assume the dictionary does not contain duplicate words.
Example 1:
1 |
|
Example 2:
1 |
|
Example 3:
1 |
|
Related Topics:Dynamic Programming
解題邏輯與實作
這題是要將題目給定的句子拆解成字典中的單字。第一個想到的就是我最愛的暴力法,拆、拆、拆就對了!
遞迴
也就是暴力解,想法很簡單,就是把句子拆成前後兩句下去查,一旦有一種拆法可以查的到就回傳 True,否則回傳 False。當然為了避免大量重複計算,例外用 HashTable 記錄查過得結果,解法如下:
- 查詢此單字是否存在於字典中
- 有,則回傳 True
- 否,則向下執行
- 使用一個指標由前向後遍歷此單字,並依指標位置,將單字拆成前後部分並呼叫遞迴傳入,唯有當前後部分皆可在字典中查到時才回傳 True,否則繼續向後遍歷。
- 若遍歷完整個字串皆無法找到拆解法則回傳 False
1 |
|
但果不期然,效率有點差跑出了個 740 ms, 1.20% 的成績,摸摸鼻子改寫 DP 去。
Dynamic Programming
這題大概是難得我這幾天整理的題目中,DP 解終於不是欠著的題目了 XDDD
這邊使用了一個 dp 陣列記錄結果,其中 dp[i] 中記錄著字串 s[:i] 是否能夠拆解,接下來若是 s[i:j] 也可以進行拆解,因為已知 s[:i] 可拆解,又s[i:j] 可拆解,故 s[:j] 可拆解,因此記錄 dp[j] 為 True,按這關係推導,最後就可以判斷目標字串是否可以被拆解。
1 |
|
這效能果然好上不少,跑出了 68 ms, 15.49% 的成績。
Dynamic Programming 改進版
看到我的 code 雖然有比上次進步,但其實無條件進位也才 16% 而已,有些好奇前面 code是怎樣的,所以手賤點了前段班的 code 出來看。
發現他雖然也是用 DP 解,但他加上了點巧思,與我不同的是他是由後向前遍歷,並寫加上了檢查距離的限制,限制在字典中最長的單字長度上,檢查再多也沒有意義,因為字典就沒有這麼長的單字咩。最後跑出來成績為 36 ms, 99.65% 。
1 |
|