C++實(shí)現(xiàn)LeetCode(312.打氣球游戲)
[LeetCode] 312. Burst Balloons 打氣球游戲
Given n balloons, indexed from 0 to n-1. Each balloon is painted with a number on it represented by array nums. You are asked to burst all the balloons. If the you burst balloon iyou will get nums[left] * nums[i] * nums[right]coins. Here left and right are adjacent indices of i. After the burst, the left and right then becomes adjacent.
Find the maximum coins you can collect by bursting the balloons wisely.
Note:
- You may imagine nums[-1] = nums[n] = 1. They are not real therefore you can not burst them.
- 0 ≤ n ≤ 500, 0 ≤ nums[i] ≤ 100
Example:
Input:
[3,1,5,8]
Output:
167
Explanation:
nums = [3,1,5,8] --> [3,5,8] --> [3,8] --> [8] --> []
coins = 3*1*5 + 3*5*8 + 1*3*8 + 1*8*1 = 167
Credits:
Special thanks to @peisi for adding this problem and creating all test cases.
這道題提出了一種打氣球的游戲,每個(gè)氣球都對(duì)應(yīng)著一個(gè)數(shù)字,每次打爆一個(gè)氣球,得到的金幣數(shù)是被打爆的氣球的數(shù)字和其兩邊的氣球上的數(shù)字相乘,如果旁邊沒(méi)有氣球了,則按1算,以此類推,求能得到的最多金幣數(shù)。參見(jiàn)題目中給的例子,題意并不難理解。那么大家拿到題后,總是會(huì)習(xí)慣的先去想一下暴力破解法吧,這道題的暴力搜索將相當(dāng)?shù)膹?fù)雜,因?yàn)槊看虮粋€(gè)氣球,斷開(kāi)的地方又重新挨上,所有剩下的氣球又要重新遍歷,這使得分治法不能 work,整個(gè)的時(shí)間復(fù)雜度會(huì)相當(dāng)?shù)母?,不要指望可以通過(guò) OJ。而對(duì)于像這種求極值問(wèn)題,一般都要考慮用動(dòng)態(tài)規(guī)劃 Dynamic Programming 來(lái)做,維護(hù)一個(gè)二維動(dòng)態(tài)數(shù)組 dp,其中 dp[i][j] 表示打爆區(qū)間 [i,j] 中的所有氣球能得到的最多金幣。題目中說(shuō)明了邊界情況,當(dāng)氣球周圍沒(méi)有氣球的時(shí)候,旁邊的數(shù)字按1算,這樣可以在原數(shù)組兩邊各填充一個(gè)1,方便于計(jì)算。這道題的最難點(diǎn)就是找狀態(tài)轉(zhuǎn)移方程,還是從定義式來(lái)看,假如區(qū)間只有一個(gè)數(shù),比如 dp[i][i],那么計(jì)算起來(lái)就很簡(jiǎn)單,直接乘以周圍兩個(gè)數(shù)字即可更新。如果區(qū)間里有兩個(gè)數(shù)字,就要算兩次了,先打破第一個(gè)再打破了第二個(gè),或者先打破第二個(gè)再打破第一個(gè),比較兩種情況,其中較大值就是該區(qū)間的 dp 值。假如區(qū)間有三個(gè)數(shù)呢,比如 dp[1][3],怎么更新呢?如果先打破第一個(gè),剩下兩個(gè)怎么辦呢,難道還要分別再遍歷算一下嗎?這樣跟暴力搜索的方法有啥區(qū)別呢,還要 dp 數(shù)組有啥意思。所謂的狀態(tài)轉(zhuǎn)移,就是假設(shè)已知了其他狀態(tài),來(lái)推導(dǎo)現(xiàn)在的狀態(tài),現(xiàn)在是想知道 dp[1][3] 的值,那么如果先打破了氣球1,剩下了氣球2和3,若之前已經(jīng)計(jì)算了 dp[2][3] 的話,就可以使用其來(lái)更新 dp[1][3] 了,就是打破氣球1的得分加上 dp[2][3]。那假如先打破氣球2呢,只要之前計(jì)算了 dp[1][1] 和 dp[3][3],那么三者加起來(lái)就可以更新 dp[1][3]。同理,先打破氣球3,就用其得分加上 dp[1][2] 來(lái)更新 dp[1][3]。說(shuō)到這里,是不是感覺(jué)豁然開(kāi)朗了 ^.^
那么對(duì)于有很多數(shù)的區(qū)間 [i, j],如何來(lái)更新呢?現(xiàn)在是想知道 dp[i][j] 的值,這個(gè)區(qū)間可能比較大,但是如果知道了所有的小區(qū)間的 dp 值,然后聚沙成塔,逐步的就能推出大區(qū)間的 dp 值了。還是要遍歷這個(gè)區(qū)間內(nèi)的每個(gè)氣球,就用k來(lái)遍歷吧,k在區(qū)間 [i, j] 中,假如第k個(gè)氣球最后被打爆,那么此時(shí)區(qū)間 [i, j] 被分成了三部分,[i, k-1],[k],和 [k+1, j],只要之前更新過(guò)了 [i, k-1] 和 [k+1, j] 這兩個(gè)子區(qū)間的 dp 值,可以直接用 dp[i][k-1] 和 dp[k+1][j],那么最后被打爆的第k個(gè)氣球的得分該怎么算呢,你可能會(huì)下意識(shí)的說(shuō),就乘以周圍兩個(gè)氣球被 nums[k-1] * nums[k] * nums[k+1],但其實(shí)這樣是錯(cuò)誤的,為啥呢?dp[i][k-1] 的意義是什么呢,是打爆區(qū)間 [i, k-1] 內(nèi)所有的氣球后的最大得分,此時(shí)第 k-1 個(gè)氣球已經(jīng)不能用了,同理,第 k+1 個(gè)氣球也不能用了,相當(dāng)于區(qū)間 [i, j] 中除了第k個(gè)氣球,其他的已經(jīng)爆了,那么周圍的氣球只能是第 i-1 個(gè),和第 j+1 個(gè)了,所以得分應(yīng)為 nums[i-1] * nums[k] * nums[j+1],分析到這里,狀態(tài)轉(zhuǎn)移方程應(yīng)該已經(jīng)躍然紙上了吧,如下所示:
dp[i][j] = max(dp[i][j], nums[i - 1] * nums[k] * nums[j + 1] + dp[i][k - 1] + dp[k + 1][j]) ( i ≤ k ≤ j )
有了狀態(tài)轉(zhuǎn)移方程了,就可以寫代碼,下面就遇到本題的第二大難點(diǎn)了,區(qū)間的遍歷順序。一般來(lái)說(shuō),遍歷所有子區(qū)間的順序都是i從0到n,然后j從i到n,然后得到的 [i, j] 就是子區(qū)間。但是這道題用這種遍歷順序就不對(duì),在前面的分析中已經(jīng)說(shuō)了,這里需要先更新完所有的小區(qū)間,然后才能去更新大區(qū)間,而用這種一般的遍歷子區(qū)間的順序,會(huì)在更新完所有小區(qū)間之前就更新了大區(qū)間,從而不一定能算出正確的dp值,比如拿題目中的那個(gè)例子 [3, 1, 5, 8] 來(lái)說(shuō),一般的遍歷順序是:
[3] -> [3, 1] -> [3, 1, 5] -> [3, 1, 5, 8] -> [1] -> [1, 5] -> [1, 5, 8] -> [5] -> [5, 8] -> [8]
顯然不是我們需要的遍歷順序,正確的順序應(yīng)該是先遍歷完所有長(zhǎng)度為1的區(qū)間,再是長(zhǎng)度為2的區(qū)間,再依次累加長(zhǎng)度,直到最后才遍歷整個(gè)區(qū)間:
[3] -> [1] -> [5] -> [8] -> [3, 1] -> [1, 5] -> [5, 8] -> [3, 1, 5] -> [1, 5, 8] -> [3, 1, 5, 8]
這里其實(shí)只是更新了 dp 數(shù)組的右上三角區(qū)域,最終要返回的值存在 dp[1][n] 中,其中n是兩端添加1之前數(shù)組 nums 的個(gè)數(shù)。參見(jiàn)代碼如下:
解法一:
class Solution { public: int maxCoins(vector<int>& nums) { int n = nums.size(); nums.insert(nums.begin(), 1); nums.push_back(1); vector<vector<int>> dp(n + 2, vector<int>(n + 2, 0)); for (int len = 1; len <= n; ++len) { for (int i = 1; i <= n - len + 1; ++i) { int j = i + len - 1; for (int k = i; k <= j; ++k) { dp[i][j] = max(dp[i][j], nums[i - 1] * nums[k] * nums[j + 1] + dp[i][k - 1] + dp[k + 1][j]); } } } return dp[1][n]; } };
對(duì)于題目中的例子[3, 1, 5, 8],得到的dp數(shù)組如下:
0 0 0 0 0 0
0 3 30 159 167 0
0 0 15 135 159 0
0 0 0 40 48 0
0 0 0 0 40 0
0 0 0 0 0 0
這題還有遞歸解法,思路都一樣,就是寫法略有不同,參見(jiàn)代碼如下:
解法二:
class Solution { public: int maxCoins(vector<int>& nums) { int n = nums.size(); nums.insert(nums.begin(), 1); nums.push_back(1); vector<vector<int>> dp(n + 2, vector<int>(n + 2, 0)); return burst(nums, dp, 1 , n); } int burst(vector<int>& nums, vector<vector<int>>& dp, int i, int j) { if (i > j) return 0; if (dp[i][j] > 0) return dp[i][j]; int res = 0; for (int k = i; k <= j; ++k) { res = max(res, nums[i - 1] * nums[k] * nums[j + 1] + burst(nums, dp, i, k - 1) + burst(nums, dp, k + 1, j)); } dp[i][j] = res; return res; } };
到此這篇關(guān)于C++實(shí)現(xiàn)LeetCode(312.打氣球游戲)的文章就介紹到這了,更多相關(guān)C++實(shí)現(xiàn)打氣球游戲內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- C++實(shí)現(xiàn)LeetCode(99.復(fù)原二叉搜索樹(shù))
- C++實(shí)現(xiàn)LeetCode(98.驗(yàn)證二叉搜索樹(shù))
- C++實(shí)現(xiàn)LeetCode(139.拆分詞句)
- C++實(shí)現(xiàn)LeetCode(97.交織相錯(cuò)的字符串)
- C++實(shí)現(xiàn)LeetCode(241.添加括號(hào)的不同方式)
- C++實(shí)現(xiàn)LeetCode(96.獨(dú)一無(wú)二的二叉搜索樹(shù))
- C++實(shí)現(xiàn)LeetCode(95.獨(dú)一無(wú)二的二叉搜索樹(shù)之二)
- C++實(shí)現(xiàn)LeetCode(104.二叉樹(shù)的最大深度)
相關(guān)文章
C和C++中的基本數(shù)據(jù)類型的大小及表示范圍詳解
這篇文章主要介紹了C和C++中的基本數(shù)據(jù)類型的大小及表示范圍詳解,基本數(shù)據(jù)類型有int、long、long long、float、double、char、string,正文有詳細(xì)介紹,歡迎參考2018-01-01C++動(dòng)態(tài)數(shù)組類的封裝實(shí)例
這篇文章主要介紹了C++動(dòng)態(tài)數(shù)組類的封裝,很重要的概念,需要的朋友可以參考下2014-08-08用C語(yǔ)言winform編寫滲透測(cè)試工具實(shí)現(xiàn)SQL注入功能
本篇文章主要介紹使用C#winform編寫滲透測(cè)試工具,實(shí)現(xiàn)SQL注入的功能。使用python編寫SQL注入腳本,基于get顯錯(cuò)注入的方式進(jìn)行數(shù)據(jù)庫(kù)的識(shí)別、獲取表名、獲取字段名,最終獲取用戶名和密碼;使用C#winform編寫windows客戶端軟件調(diào)用.py腳本,實(shí)現(xiàn)用戶名和密碼的獲取2021-08-08C++11中value category(值類別)及move semantics(移動(dòng)語(yǔ)義)的介紹
這篇文章主要給大家介紹了C++11中value category(值類別)及move semantics(移動(dòng)語(yǔ)義)的介紹,文中介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2018-05-05C語(yǔ)言驅(qū)動(dòng)開(kāi)發(fā)之內(nèi)核通過(guò)PEB獲取進(jìn)程參數(shù)
PEB結(jié)構(gòu)(Process Envirorment Block Structure)其中文名是進(jìn)程環(huán)境塊信息。本文將通過(guò)PEB實(shí)現(xiàn)獲取進(jìn)程參數(shù),感興趣的小伙伴可以了解一下2022-10-10Linux中使用C語(yǔ)言的fork()函數(shù)創(chuàng)建子進(jìn)程的實(shí)例教程
fork是一個(gè)在Linux系統(tǒng)環(huán)境下專有的函數(shù),現(xiàn)有的進(jìn)程調(diào)用fork后將會(huì)創(chuàng)建一個(gè)新的進(jìn)程,這里我們就來(lái)看一下Linux中使用C語(yǔ)言的fork()函數(shù)創(chuàng)建子進(jìn)程的實(shí)例教程2016-06-06