C++动态规划dp算法题

看我主页简介免费C++学习资源,视频教程、职业规划、面试详解、学习路线、开发工具

每晚8点直播讲解C++编程技术。

问题1:找硬币,换钱的方法

输入:

penny数组代表所有货币的面值,正数不重复

aim小于等于1000,代表要找的钱

输出:

换钱的方法总数

解法1:经典dp,空间复杂度O(n*aim)

class Exchange {

public:

int countWays(vector<int> penny, int n, int aim) {

if (penny.empty()||n == 0)

return 0;

vector<vector<int> > dp(n,vector<int>(aim+1)); //二维数组dp

for (int i = 0;i < n;i++) {

dp[i][0] = 1;

}

for (int j = 1;j < aim+1;j++) {

dp[0][j] = j%penny[0] == 0?1:0;  //只需要算dp[0][j]

for (int i = 1;i < n;i++) {

for (int j = 1;j < aim+1;j++) {

dp[i][j] = (j-penny[i]) >= 0?(dp[i-1][j] + dp[i][j-penny[i]]):dp[i-1][j];  //这是关键,不用管penny【i】到底使用了几次,直接减去1次使用就好               

}

}

return dp[n-1][aim];

}

};

解法2:与上面的问题一样,只不过在求dp时只使用1维数组来做;使用迭代,时间复杂度一样:

class Exchange {

public:

int countWays(vector<int> penny, int n, int aim) {

vector<int> dp(aim + 1);

for (int i = 0; i <= aim; i++)

if (i % penny[0] == 0)

dp[i] = 1;

for (int i = 1; i < n; i++)

for (int j = 1; j <= aim; j++)

if ( j >= penny[i]) //条件,如果不满足就直接等于上轮的结果,不用做修改

dp[j] += dp[j - penny[i]];

return dp[aim];

}

};

问题2:跳台阶问题:

其实是斐波那契问题,f(n)=f(n-1)+f(n-2)

#include <iostream>

using namespace std;

int main(){

int step;

while(cin>>step){

vector<int> dp(2,1); //初始化赋值

dp[1]=2;

int temp;

for(int i=3;i<=step;i++){

temp=dp[0];

dp[0]=dp[1];

dp[1]=dp[1]+temp;

}

if(step==1) dp[1]=1;;

cout<<dp[1]<<endl;

}

return 0;

}

问题3:走矩阵,求路劲最小和,或者是求整个路径

n×m的map,则 f(n,m)=min(f(n-1,m),f(n,m-1))+map[n][m];

由于这里和问题1类似,可以只用到一个一维数组求解;

class MinimumPath {

public:

int getMin(vector<vector<int> > map, int n, int m) {

vector<int> dp(m,0);

dp[0] = map[0][0];

for (int i = 1,j = 0;i < m;i++,j++) {

dp[i] = map[0][i]+dp[j];

}

for (int i = 1;i < n;i++) {

dp[0] += map[i][0];    //不能忘了dp[0]的更新

for (int j = 1;j < m;j++) {

dp[j] = min(dp[j],dp[j-1])+map[i][j]; //如果求路径,则在这里记录,需要额外存储空间

}

}

return dp[m-1];

}

};

问题4:最长上升子序列问题(LIS)

解法:O(N方)用dp数组的dp[i]记录下以A[i]结尾的递增子序列中最长的长度,计算dp[i+1]时,遍历A[0~i]找到比A[i+1]小的元素,再比较与这些元素对应的dp数组中的值,找到最大的一个再加1,赋值给dp[i+1]。

class LongestIncreasingSubsequence {

public:

int getLIS(vector<int> A, int n) {

if (A.empty()||n == 0)

return 0; 

vector<int> dp(n,0);

dp[0] = 1;

int resMax = 0;

for (int i = 1;i < n;i++) {

int tempMax = 0;

for (int j = 0;j < i;j++) {

if (A[i] > A[j])

tempMax = max(tempMax,dp[j]);

}

dp[i] = ++tempMax;

resMax = max(resMax,dp[i]);  //记录最大的上升子序列长度,因为当前i可能并不在最长上升子序列中

}

return resMax;

}

};

如上的实现复杂度为N方,可以增加归纳的假设,增加b[k]存储长度为k最长子序列最小结尾元素,那么可以利用二分查找,使用logn查找到插入点,对于每次比较,要么直接比较b【k】比它大直接k+1,更新b【k+1】,要么二分查找到位置,更新b【j】,所以最终复杂度为nlogn(如果数据量大的话,使用该算法较好)

问题5:最长公共子序列长度(LCS)

上图可以看出使用了斜侧的比较,所以不能再使用1维数组了

class LCS {

public:

int findLCS(string A, int n, string B, int m) {

if (A.empty()||n==0||B.empty()||m==0)

return 0;

vector<vector<int> > dp(n,vector<int>(m));

//下面是两个for的初始化,当出现第一个相等的时,后面的都直接赋值为1;

for (int i = 0;i < m;i++) {

if (A[0] == B[i]) {

for (int j = i;j < m;j++)

dp[0][j] = 1;

break ;

}             

}

for (int i = 0;i < n;i++) {

if (B[0] == A[i]) {

for (int j = i;j < n;j++)

dp[j][0] = 1;

break ;

}             

}

for (int i = 1;i < n;i++) {

for (int j = 1;j < m;j++) {

if (A[i] == B[j])

dp[i][j] = dp[i-1][j-1]+1;

else

dp[i][j] = max(dp[i-1][j],dp[i][j-1]);

}

}

return dp[n-1][m-1];

}

};

上面的方法中初始化第一行和第一列有点麻烦,增加了额外的语句,可以增加数组一行和一列来优化代码:

class LCS {

public:

int findLCS(string A, int n, string B, int m) {       

vector<vector<int> > dp(n+1,vector<int>(m+1,0));

for (int i =1;i<=n ;++i){           

for (int j=1; j<=m; ++j){

if (A[i-1] == B[j-1]){

dp[i][j]  = dp[i-1][j-1]+1; //第1行也可以照此直接初始化

}

else {

dp[i][j] = max( dp[i-1][j] ,dp[i][j-1]);

}               

}

}

return dp[n][m];

}

};

问题6:背包

N件物品,价值记录在数组V,重量记录在数组W,背包总重量最大为cap,要求总价值最大;

class Backpack {

public:

int maxValue(vector<int> w, vector<int> v, int n, int cap) {

if (w.empty()||v.empty()||n==0||cap==0)

return 0;

vector<vector<int> > dp(n,vector<int>(cap+1));

for (int j = 1;j < cap+1;j++) {

dp[0][j] = w[0] <= j?v[0]:0;

}

for (int i = 0;i < n;i++) {

dp[i][0] = 0;

}

for (int i = 1;i < n;i++) {

for (int j = 1;j < cap+1;j++) {

if (w[i] > j)

dp[i][j] = dp[i-1][j];

else

dp[i][j] = max(dp[i-1][j],v[i]+dp[i-1][j-w[i]]); //由于该问题每个物品最多只能放1件,如果不限制个数的话,则在这里修改条件

}

}

return dp[n-1][cap];

}

};

由于没有用到斜侧的比较,所以可以使用1维的数组:

class Backpack {

public:

int maxValue(vector<int> w, vector<int> v, int n, int cap) {

if (w.empty()||v.empty()||n==0||cap==0)

return 0;

vector<int> dp(cap+1,0);       

for (int i = 0;i < n;i++) {

vector<int> last(dp);

for (int j = 1;j < cap+1;j++) {

dp[j] = j < w[i]?last[j]:max(last[j],v[i]+last[j-w[i]]);               

}

}

return dp[cap];

}

};

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,980评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,178评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,868评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,498评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,492评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,521评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,910评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,569评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,793评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,559评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,639评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,342评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,931评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,904评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,144评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,833评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,350评论 2 342

推荐阅读更多精彩内容

  • 在C语言中,五种基本数据类型存储空间长度的排列顺序是: A)char B)char=int<=float C)ch...
    夏天再来阅读 3,318评论 0 2
  • 计算机二级C语言上机题库(南开版) 1.m个人的成绩存放在score数组中,请编写函数fun,它的功能是:将低于平...
    MrSunbeam阅读 6,304评论 1 42
  • 回溯算法 回溯法:也称为试探法,它并不考虑问题规模的大小,而是从问题的最明显的最小规模开始逐步求解出可能的答案,并...
    fredal阅读 13,620评论 0 89
  • 【程序1】 题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔...
    开心的锣鼓阅读 3,303评论 0 9
  • 大约这般,在下雨的日子里,才有了哀伤的小模样。 一,他不喜欢你 二,是你想太多 三,人生呐,真是寂寞如雪
    我若彩虹阅读 249评论 0 0