2.3 記錄結(jié)果再利用的“動態(tài)規(guī)劃”

2.3.1 記憶化搜索與動態(tài)規(guī)劃

1. 01背包問題

<pre><code>
//
// Created by Nathan on 15/3/23.
// Copyright (c) 2015年 Nathan. All rights reserved.
//
// 1<=n<=100;
// 1<=wi,vi<=100;
// 1<=W<=10000;
// n=4;
// (w,v) ={(2,3),(1,2),(3,4),(2,2)}
// W=5;
//

include <iostream>

using namespace std;
int n,W;
const int MAX = 100;
int w[MAX],v[MAX];
int solve(){
int MAX_N = 100, MAX_W = 10000;
int dp[MAX_N+1][MAX_W+1];
for (int i =0 ; i<n; i++) {
for (int j=0; j<=W; j++) {
if(j>=w[i]){
dp[i+1][j] = max( dp[i][j],dp[i][j-w[i]]+v[i]);
} else {
dp[i+1][j] = dp[i][j];
}
}
}
return dp[n][W];
}
int solve_single(){
int MAX_W = 10000;
int dp[MAX_W];
for (int i=0; i<n; i++) {
for (int j = W; j >= 0; j--) {
if( j>=w[i] ){
dp[j] = max(dp[j], dp[j-w[i]]+v[i] );
}
}
}
return dp[W];
}
int main(int argc, const char * argv[]) {
cin >> n >> W;
for (int i = 0; i < n; i++) {
cin >> w[i];
}
for (int i = 0; i < n; i++) {
cin >> v[i];
}
cout << "Solve_singe:" << solve_single() <<endl;
cout << "Solve:" << solve() <<endl;
return 0;
}
</code></pre>

2. 最長公共子序列問題

<pre><code>
//
// Created by Nathan on 15/3/25.
// Copyright (c) 2015年 Nathan. All rights reserved.
//
// 1 <=n,m <=1000
// n = 4;
// m = 4;
// s = 'abcd';
// t = 'becd';
//

include <iostream>

using namespace std;
const int MAX = 1000;
int dp[MAX+1][MAX+1];
int n,m;
char s[MAX],t[MAX];
int solve(){
for (int i = 0; i<n; i++) {
for (int j = 0; j<m; j++) {
if(s[i]==t[j]){
dp[i+1][j+1] = dp[i][j]+1;
} else {
dp[i+1][j+1] = max(dp[i+1][j], dp[i][j+1]);
}
}
}
return dp[n][m];
}
int main(int argc, const char * argv[]) {
cin >> n >> m;
for (int i = 0; i < n; i++) {
cin >> s[i];
}
for (int i = 0; i < m; i++) {
cin >> t[i];
}
cout << solve() << endl;
return 0;
}
</code></pre>

2.3.2 進一步探討遞推關(guān)系

1. 完全背包問題1

<pre><code>
//
// Created by Nathan on 15/3/27.
// Copyright (c) 2015年 Nathan. All rights reserved.
// 完全背包1
// 每件物品可以挑選任意多件
// 1 <= n <= 100
// 1 <= wi,vi <= 100
// 1 <= W <= 10000
// n = 3
// (w,v) = {(3,4),(4,5),(2,3)}
// W = 7
//

include <iostream>

using namespace std;
const int MAX = 100;
const int MAX_R = 10000;
int n,W;
int w[MAX],v[MAX];
int solve_single(){
int dp[MAX_R];
for (int i = 0; i<n; i++) {
for (int j=w[i]; j<=W; j++) {
dp[j] = max(dp[j],dp[j-w[i]]+v[i] );
}
}
return dp[W];
}
int solve(){
int dp[MAX+1][MAX_R+1];
for (int i = 0; i<n; i++) {
for (int j = 0; j<=W; j++) {
if( j>=w[i] ){
dp[i+1][j] = max(dp[i][j], dp[i+1][j-w[i]]+v[i]);
} else {
dp[i+1][j] = dp[i][j];
}
}
}
return dp[n][W];
}
int main(int argc, const char * argv[]) {
cin >> n >> W;
for (int i =0; i<n; i++) {
cin >> w[i];
}
for (int i =0; i<n; i++) {
cin >> v[i];
}
cout << "Solve:" << solve() << endl;
cout << "Solve_single:" << solve_single() << endl;
return 0;
}
</code></pre>

2. 完全背包問題2

<pre><code>
//
// main.cpp
// 2.3.2.2 Package_150328
//
// Created by Nathan on 15/3/28.
// Copyright (c) 2015年 Nathan. All rights reserved.
// 1<= n <= 100;
// 1<= wi <= 10000000;
// 1<= vi <= 100;
// 1<= W <= 1000000000;
// n = 4
// (w,v) = {(2,3),(1,2),(3,4),(2,2)}
// W = 5
//

include <iostream>

using namespace std;
int n,W;
const int MAX_N = 100;
const int MAX_V = 100;
const int MAX_W = 10000000;
const int INF = 1000000;
int w[MAX_W],v[MAX_V],dp[MAX_N+1][MAX_NMAX_V+1];
int solve(){
fill(dp[0], dp[0]+MAX_N
MAX_V+1, INF);
dp[0][0]=0;
for (int i = 0; i<n; i++) {
for (int j = 0 ; j<=MAX_NMAX_V; j++) {
if(j<v[i]){
dp[i+1][j] = dp[i][j];
} else {
dp[i+1][j] = min(dp[i][j], dp[i][j-v[i]]+w[i]);
}
}
}
int res = 0;
for (int i =0 ; i<=MAX_N
MAX_V; i++) {
if(dp[n][i]<=W){
res = i;
}
}
return res;
}
int main(int argc, const char * argv[]) {
cin >> n >> W;
for (int i = 0; i < n; i++) {
cin >> w[i];
}
for (int i = 0; i < n; i++) {
cin >> v[i];
}
cout << solve() << endl;
return 0;
}
</code></pre>

3. 多重部分和問題

<pre><code>
//
// main.cpp
// 2.3.2.3 Multi-section_150328
//
// Created by Nathan on 15/3/28.
// Copyright (c) 2015年 Nathan. All rights reserved.
// 3 17
// 3 5 8
// 3 2 2
//

include <iostream>

using namespace std;
int n,K;
const int MAX_N = 100;
const int MAX_K = 100000;
int a[MAX_N],m[MAX_N];
int dp[MAX_N+1][MAX_K+1];
int solve(){
dp[0][0]=true;
for (int i=0; i<n; i++) {
for (int j=0; j<=K; j++) {
for (int k =0; ka[i]<=j&&k<=m[i]; k++) {
dp[i+1][j] |= dp[i][j-k
a[i]];
}
}
}
for (int i=0; i<=n; i++) {
for (int j=0; j<=K; j++) {
cout << dp[i][j] << " ";
}
cout << endl;
}
return dp[n][K];
}
int main(int argc, const char * argv[]) {
cin >> n >> K;
for (int i = 0; i<n; i++) {
cin >> a[i];
}
for (int i = 0; i<n; i++) {
cin >> m[i];
}
int res = solve();
cout << res <<endl;
return 0;
}
</code></pre>

4.最長上升子序列

<pre><code>
//
// main.cpp
// 2.3.2.4 LIS_150328
//
// Created by Nathan on 15/3/28.
// Copyright (c) 2015年 Nathan. All rights reserved.
//

include <iostream>

using namespace std;
const int MAX = 1000;
int n,a[1000000];
int dp[MAX];
const int INF = 100000000;
int solve_sec(){
fill(dp, dp+n, INF);
for (int i = 0; i<n; i++) {
*lower_bound(dp, dp+n, a[i])=a[i];
}
cout << lower_bound(dp, dp+n, INF)-dp;
return 0;
}
int solve(){
int res = 0;
for (int i =0; i<n; i++) {
dp[i]=1;
for (int j=0; j<i; j++) {
if (a[j]<a[i]) {
dp[i]=max( dp[i], dp[j]+1);
}
}
res = max( res , dp[i]);
for (int k=0; k<n; k++) {
cout << dp[k] << " " ;
}
cout << endl;
}
return res;
}
int main(int argc, const char * argv[]) {
cin >> n;
for (int i = 0; i<n; i++) {
cin >> a[i];
}
//solve_sec();
//int res = solve_sec();
cout << solve() << endl;
return 0;
}
</code></pre>

5. Lower_bound

lower_bound是STL(標準模板庫)中的一個函數(shù)。
<code>lower_bound(<#_ForwardIterator __first#>, <#_ForwardIterator __last#>, <#const _Tp &_value#>)</code>
該函數(shù)的作用是從已拍好的序列a中利用二分搜索找出指向滿足ai≥k的ai指針傅事。類似的upper_bound是找出滿足ai>k的ai的最小指針绵载。
<pre><code>

include <iostream>

include <algorithm>

include <vector>

using namespace std;
int main () {
int myints[] = {10,20,30,30,20,10,10,20};
vector<int> v(myints,myints+8); // 10 20 30 30 20 10 10 20
sort (v.begin(), v.end()); // 10 10 10 20 20 20 30 30
vector<int>::iterator low,up;
low=lower_bound (v.begin(), v.end(), 10); // ^
up= upper_bound (v.begin(), v.end(), 10); // ^
cout << "lower_bound at position " << (low- v.begin()) << endl;
cout << "upper_bound at position " << (up - v.begin()) << endl;
return 0;
}
</code></pre>

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末蜕猫,一起剝皮案震驚了整個濱河市垮卓,隨后出現(xiàn)的幾起案子傻工,更是在濱河造成了極大的恐慌挑格,老刑警劉巖咙冗,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異漂彤,居然都是意外死亡雾消,警方通過查閱死者的電腦和手機灾搏,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來立润,“玉大人狂窑,你說我怎么就攤上這事∩H” “怎么了泉哈?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長破讨。 經(jīng)常有香客問我丛晦,道長,這世上最難降的妖魔是什么添忘? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任采呐,我火速辦了婚禮,結(jié)果婚禮上搁骑,老公的妹妹穿的比我還像新娘斧吐。我一直安慰自己,他們只是感情好仲器,可當我...
    茶點故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布煤率。 她就那樣靜靜地躺著,像睡著了一般乏冀。 火紅的嫁衣襯著肌膚如雪蝶糯。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天辆沦,我揣著相機與錄音昼捍,去河邊找鬼。 笑死肢扯,一個胖子當著我的面吹牛妒茬,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播蔚晨,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼乍钻,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了铭腕?” 一聲冷哼從身側(cè)響起银择,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎累舷,沒想到半個月后浩考,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡被盈,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年怀挠,在試婚紗的時候發(fā)現(xiàn)自己被綠了析蝴。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,711評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡绿淋,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出尝盼,到底是詐尸還是另有隱情吞滞,我是刑警寧澤,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布盾沫,位于F島的核電站裁赠,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏赴精。R本人自食惡果不足惜佩捞,卻給世界環(huán)境...
    茶點故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望蕾哟。 院中可真熱鬧一忱,春花似錦、人聲如沸谭确。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽逐哈。三九已至芬迄,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間昂秃,已是汗流浹背禀梳。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留肠骆,地道東北人算途。 一個月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像哗戈,于是被迫代替她去往敵國和親郊艘。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 44,611評論 2 353

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

  • 簡介 搜索迷宮(BFS+隊列) 最短路Dijkstra+鄰接矩陣Dijkstra+鏈式前向星+優(yōu)先隊列Bellma...
    染微言閱讀 397評論 0 1
  • 動態(tài)規(guī)劃(Dynamic Programming) 本文包括: 動態(tài)規(guī)劃定義 狀態(tài)轉(zhuǎn)移方程 動態(tài)規(guī)劃算法步驟 最長...
    廖少少閱讀 3,279評論 0 18
  • 回溯算法 回溯法:也稱為試探法,它并不考慮問題規(guī)模的大小讶凉,而是從問題的最明顯的最小規(guī)模開始逐步求解出可能的答案狸棍,并...
    fredal閱讀 13,650評論 0 89
  • 定義一組子問題,按照由小到大狞贱,以小問題的解答支持大問題求解的模式,依次解決所有的子問題蜀涨,并最終得到原問題的解答瞎嬉。 ...
    芥丶未央閱讀 866評論 0 2
  • CSS Hack 由于不同廠商的流覽器或某瀏覽器的不同版本(如IE6-IE11,Firefox/Safari/Op...
    shawnzx閱讀 615評論 0 50