【C++刷题】优选算法——动态规划第一辑

2024-03-17 04:28

本文主要是介绍【C++刷题】优选算法——动态规划第一辑,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.状态表示是什么?简答理解是dp表里的值所表示的含义怎么来的?题目要求经验+题目要求分析问题的过程中,发现重复子问题
2.状态转移方程dp[i]=......细节问题:3.初始化控制填表的时候不越界4.填表顺序控制在填写当前状态的时候,所需要的状态已经填写好了5.返回值题目要求+状态表示空间优化滚动数组
  1. 第 N 个泰波那契数
int tribonacci(int n)
{// 处理一些边界情况if(n < 3){if(n == 0) return 0;else return 1;}// 1.创建dp表vector<int> dp(n + 1);// 2.初始化dp[0] = 0, dp[1] = 1, dp[2] = 1;for(int i = 3; i <= n; ++i){// 3.填表dp[i] = dp[i - 1] + dp[i - 2] + dp[i - 3];}// 4.返回值return dp[n];
}
// 空间优化版本
int tribonacci(int n)
{int arr[3] = { 0,1,1 };if(n < 3) return arr[n];int ret = 0;for(int i = 3; i <= n; ++i){ret = arr[0] + arr[1] + arr[2];arr[0] = arr[1], arr[1] = arr[2], arr[2] = ret;}return ret;
}
  1. 三步问题
状态表示:经验+题目要求:以i位置为结尾来入手dp[i]: 表示到达i位置,一共有多少种方法
状态转移方程:基于i位置状态,跨一步到i位置,来划分问题
int waysToStep(int n)
{if(1 == n) return 1;else if(2 == n) return 2;else if(3 == n) return 4;// 1.dp数组vector<int> dp(n + 1);// 2.初始化dp[1] = 1, dp[2] = 2, dp[3] = 4;for(int i = 4; i <= n; ++i){// 3.状态方程dp[i] = ((dp[i - 1] + dp[i - 2]) % 1000000007 + dp[i - 3]) % 1000000007;}// 4.返回值return dp[n];
}
  1. 使用最小花费爬楼梯
状态表示:经验+题目要求:以i位置为结尾来入手dp[i]: 表示i位置到下一步的最小花费
状态转移方程:dp[i] = min(dp[i-1], dp[i-2]) + cost[i]
int minCostClimbingStairs(vector<int>& cost)
{// 1.dp数组vector<int> dp(cost.size());// 2.初始化dp[0] = cost[0]; dp[1] = cost[1];for (int i = 2; i < dp.size(); ++i){// 3.状态转移方程dp[i] = min(dp[i - 1], dp[i - 2]) + cost[i];}// 4.返回值return min(dp[dp.size() - 1], dp[dp.size() - 2]);
}
  1. 解码方法
状态表示:经验+题目要求:以i位置为结尾来入手dp[i]: 表示以i位置为结尾时,解码方法的总数
状态转移方程:

在这里插入图片描述

int numDecodings(string s)
{// 0.边界情况if(s.size() < 2){if(s[0] == '0') return 0;else return 1;}// 1.dp数组vector<int> dp(s.size(), 0);// 2.初始化if (s[0] == '0') dp[0] = 0;else dp[0] = 1;if (s[0] != '0' && s[1] != '0') dp[1] += 1;if (10 <= stoi(s.substr(0, 2)) && stoi(s.substr(0, 2)) <= 26) dp[1] += 1;for(int i = 2; i < dp.size(); ++i){// 3.状态转移方程int num1 =0, num2 = 0;if(s[i] != '0') num1 = dp[i - 1];if(10 <= stoi(s.substr(i - 1, 2)) && stoi(s.substr(i - 1, 2)) <= 26) num2 = dp[i - 2];dp[i] = num1 + num2;}// 4.返回值return dp.back();
}
  1. 不同路径
状态表示:经验+题目要求:以[i,j]位置为结尾来入手dp[i][j]: 表示以[i,j]位置为finish时,从start出发的不同路径数
状态转移方程:dp[i][j] = dp[i-1][j] + dp[i][j-1]
int uniquePaths(int m, int n)
{// 1.dp数组vector<vector<int>> dp(m, vector<int>(n));// 2.初始化for (int i = 0; i < m; ++i){dp[i][0] = 1;}for (int i = 0; i < n; ++i){dp[0][i] = 1;}// 3.状态转移方程for (int row = 1; row < m; ++row){for (int col = 1; col < n; ++col){dp[row][col] = dp[row - 1][col] + dp[row][col - 1];}}// 4.返回值return dp.back().back();
}
  1. 不同路径 II
int uniquePathsWithObstacles(vector<vector<int>>& obstacleGrid)
{// 1.dp数组int m = obstacleGrid.size();int n = obstacleGrid[0].size();vector<vector<int>> dp(m, vector<int>(n));// 2.初始化for(int i = 0; i < m; ++i){if(obstacleGrid[i][0] == 1)break;dp[i][0] = 1;}for(int i = 0; i < n; ++i){if(obstacleGrid[0][i] == 1)break;dp[0][i] = 1;}// 3.状态转移方程for(int row = 1; row < m; ++row){for(int col = 1; col < n; ++col){if(obstacleGrid[row][col] == 1)continue;dp[row][col] = dp[row - 1][col] + dp[row][col - 1];}}// 4.返回值return dp.back().back();
}
  1. 珠宝的最高价值
状态表示:经验+题目要求:以[i,j]位置为结尾来入手dp[i][j]: 表示到达[i,j]位置时所能得到的的最大价值
状态转移方程:dp[i][j] = max(dp[i-1][j], dp[i][j-1]) + frame[i][j]
int jewelleryValue(vector<vector<int>>& frame)
{// 1.dp数组int row = frame.size();int col = frame[0].size();vector<vector<int>> dp(row, vector<int>(col));// 2.初始化dp[0][0] = frame[0][0];for(int i = 1; i < col; ++i){dp[0][i] = dp[0][i - 1] + frame[0][i];}for(int i = 1; i < row; ++i){dp[i][0] = dp[i - 1][0] + frame[i][0];}// 3.状态转移方程for(int i = 1; i < row; ++i){for(int j = 1; j < col; ++j){dp[i][j] = max(dp[i - 1][j], dp[i][j - 1]) + frame[i][j];}}// 4.返回值return dp.back().back();
}
  1. 下降路径最小和
状态表示:经验+题目要求:以[i,j]位置为结尾来入手dp[i][j]: 表示到达[i,j]位置时所得到的最小下降路径和
状态转移方程:dp[i][j] = min(dp[i-1][j-1], dp[i-1][j], dp[i-1][j+1]) + frame[i][j]
    int minFallingPathSum(vector<vector<int>>& matrix){// 1.dp数组int n = matrix.size();vector<vector<int>> dp(n, vector<int>(n));// 2.初始化for(int i = 0; i < n; ++i){dp[0][i] = matrix[0][i];}// 3.状态转移方程for(int i = 1; i < n; ++i){for(int j = 0; j < n; ++j){if(j == 0){dp[i][j] = min(dp[i - 1][j], dp[i - 1][j + 1]) + matrix[i][j];}else if(j == n - 1){dp[i][j] = min(dp[i - 1][j], dp[i - 1][j - 1]) + matrix[i][j];}else{dp[i][j] = min(min(dp[i - 1][j - 1], dp[i - 1][j]), dp[i - 1][j + 1]) + matrix[i][j];}}}// 4.返回值int min_sum = dp[n - 1][0];for(int i = 1; i < n; ++i){if(dp[n - 1][i] < min_sum) min_sum = dp[n - 1][i];}return min_sum;}
  1. 最小路径和
状态表示:经验+题目要求:以[i,j]位置为结尾来入手dp[i][j]: 表示到达[i,j]位置时所得到的最小路径和
状态转移方程:dp[i][j] = min(dp[i-1][j], dp[i][j-1]) + grid[i][j]
int minPathSum(vector<vector<int>>& grid)
{// 1.dp数组int m = grid.size();int n = grid[0].size();vector<vector<int>> dp(m, vector<int>(n));// 2.初始化dp[0][0] = grid[0][0];for(int i = 1; i < m; ++i){dp[i][0] = dp[i - 1][0] + grid[i][0];}for(int i = 1; i < n; ++i){dp[0][i] = dp[0][i - 1] + grid[0][i];}// 3.状态转移方程for(int i = 1; i < m; ++i){for(int j = 1; j < n; ++j){dp[i][j] = min(dp[i - 1][j], dp[i][j - 1]) + grid[i][j];}}// 4.返回值return dp.back().back();
}
  1. 地下城游戏
状态表示:经验+题目要求:以[i,j]位置为起点来入手dp[i][j]: 表示从[i,j]位置出发,到达终点,所需的最低初始健康点数
状态转移方程:dp[i][j] = min(dp[i + 1][j], dp[i][j + 1]) - dungeon[i][j];dp[i][j] = max(1, dp[i][j]); // 细节处理,健康点数至少为1才能存活
int calculateMinimumHP(vector<vector<int>>& dungeon)
{// 1.dp数组int m = dungeon.size();int n = dungeon[0].size();vector<vector<int>> dp(m, vector<int>(n));// 2.初始化if(dungeon[m - 1][n - 1] < 0) dp[m - 1][n - 1] = 1 - dungeon[m - 1][n - 1];else dp[m - 1][n - 1] = 1;for(int i = n - 2; i >= 0; --i){dp[m - 1][i] = dp[m - 1][i + 1] - dungeon[m - 1][i];dp[m - 1][i] = max(1, dp[m - 1][i]);}for(int i = m - 2; i >= 0; --i){dp[i][n - 1] = dp[i + 1][n - 1] - dungeon[i][n - 1];dp[i][n - 1] = max(1, dp[i][n - 1]);}// 3.状态转移方程for(int i = m - 2; i >= 0; --i){for(int j = n - 2; j >= 0; --j){dp[i][j] = min(dp[i + 1][j], dp[i][j + 1]) - dungeon[i][j];dp[i][j] = max(1, dp[i][j]);}}// 4.返回值return dp[0][0];
}

这篇关于【C++刷题】优选算法——动态规划第一辑的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/817788

相关文章

Windows下C++使用SQLitede的操作过程

《Windows下C++使用SQLitede的操作过程》本文介绍了Windows下C++使用SQLite的安装配置、CppSQLite库封装优势、核心功能(如数据库连接、事务管理)、跨平台支持及性能优... 目录Windows下C++使用SQLite1、安装2、代码示例CppSQLite:C++轻松操作SQ

C++中RAII资源获取即初始化

《C++中RAII资源获取即初始化》RAII通过构造/析构自动管理资源生命周期,确保安全释放,本文就来介绍一下C++中的RAII技术及其应用,具有一定的参考价值,感兴趣的可以了解一下... 目录一、核心原理与机制二、标准库中的RAII实现三、自定义RAII类设计原则四、常见应用场景1. 内存管理2. 文件操

C++中零拷贝的多种实现方式

《C++中零拷贝的多种实现方式》本文主要介绍了C++中零拷贝的实现示例,旨在在减少数据在内存中的不必要复制,从而提高程序性能、降低内存使用并减少CPU消耗,零拷贝技术通过多种方式实现,下面就来了解一下... 目录一、C++中零拷贝技术的核心概念二、std::string_view 简介三、std::stri

C++高效内存池实现减少动态分配开销的解决方案

《C++高效内存池实现减少动态分配开销的解决方案》C++动态内存分配存在系统调用开销、碎片化和锁竞争等性能问题,内存池通过预分配、分块管理和缓存复用解决这些问题,下面就来了解一下... 目录一、C++内存分配的性能挑战二、内存池技术的核心原理三、主流内存池实现:TCMalloc与Jemalloc1. TCM

C++ 函数 strftime 和时间格式示例详解

《C++函数strftime和时间格式示例详解》strftime是C/C++标准库中用于格式化日期和时间的函数,定义在ctime头文件中,它将tm结构体中的时间信息转换为指定格式的字符串,是处理... 目录C++ 函数 strftipythonme 详解一、函数原型二、功能描述三、格式字符串说明四、返回值五

C++作用域和标识符查找规则详解

《C++作用域和标识符查找规则详解》在C++中,作用域(Scope)和标识符查找(IdentifierLookup)是理解代码行为的重要概念,本文将详细介绍这些规则,并通过实例来说明它们的工作原理,需... 目录作用域标识符查找规则1. 普通查找(Ordinary Lookup)2. 限定查找(Qualif

Java调用C#动态库的三种方法详解

《Java调用C#动态库的三种方法详解》在这个多语言编程的时代,Java和C#就像两位才华横溢的舞者,各自在不同的舞台上展现着独特的魅力,然而,当它们携手合作时,又会碰撞出怎样绚丽的火花呢?今天,我们... 目录方法1:C++/CLI搭建桥梁——Java ↔ C# 的“翻译官”步骤1:创建C#类库(.NET

MyBatis编写嵌套子查询的动态SQL实践详解

《MyBatis编写嵌套子查询的动态SQL实践详解》在Java生态中,MyBatis作为一款优秀的ORM框架,广泛应用于数据库操作,本文将深入探讨如何在MyBatis中编写嵌套子查询的动态SQL,并结... 目录一、Myhttp://www.chinasem.cnBATis动态SQL的核心优势1. 灵活性与可

C/C++ chrono简单使用场景示例详解

《C/C++chrono简单使用场景示例详解》:本文主要介绍C/C++chrono简单使用场景示例详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友... 目录chrono使用场景举例1 输出格式化字符串chrono使用场景China编程举例1 输出格式化字符串示

C++/类与对象/默认成员函数@构造函数的用法

《C++/类与对象/默认成员函数@构造函数的用法》:本文主要介绍C++/类与对象/默认成员函数@构造函数的用法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录名词概念默认成员函数构造函数概念函数特征显示构造函数隐式构造函数总结名词概念默认构造函数:不用传参就可以