64、最小路径和

题目:

解答:

简单dp。

定义:dp[i][j]为到达(i,j)所需要的最短路程

初始化:dp[0][0]=grid[0][0],同时对第一行和第一列的,第i个就是前i个之和加上自身

递归:dp[i][j]=min(dp[i-1][j],dp[i][j-1])+grid[i][j],也就是从上面到达或者从左边到达

空间复杂度O(mn),m为grid行数,n为grid列数,作空间优化

vector<vector<int>> dp(m,vector<int>(2)) m行2列的vector 降低空间复杂度为m(这里也可以先写个if,判断m和n大小,来决定是m行2列还是m列2行,不影响时间复杂度)

按照一列一列来遍历,修改初始化条件,先初始化第一列,然后从第二列开始,dp[0][1]单独计算,dp[j][1]按照上述递归式子的算法计算。一列遍历完成后,用dp[m][0]=dp[m][1]来存储状态,继续扫描下一列。最后return dp[m-1][1]即可

cpp 复制代码
class Solution {
public:
    int minPathSum(vector<vector<int>>& grid) {
        int m = grid.size();
        int n = grid[0].size();
        if(n==1) {
            int ans = 0;
            for(int i=0;i<m;i++)    
                ans+=grid[i][0];
            return ans;
        }
        vector<vector<int>> dp(m,vector<int>(2));
        //dp[j][1]=min(dp[j][0],dp[j-1][1])+grid[j][i]
        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++){
            for(int j=0;j<m;j++){
                if(j==0)
                    dp[j][1]=grid[j][i]+dp[j][0];
                else
                    dp[j][1]=min(dp[j-1][1],dp[j][0])+grid[j][i];  
            }
            for(int j=0;j<m;j++)
                dp[j][0]=dp[j][1];
        }
        return dp[m-1][1];
    }
};

时间复杂度O(mn)

空间复杂度O(m)

相关推荐
码破苍穹ovo2 小时前
堆----1.数组中的第K个最大元素
java·数据结构·算法·排序算法
愤怒的小鸟~~~2 小时前
c语言创建的一个队列结构(含有这个头指针和这个尾指针的结构具有一定的参考价值)
c语言·开发语言·算法
Joker-01114 小时前
深入 Go 底层原理(十二):map 的实现与哈希冲突
算法·go·哈希算法·map
金融小师妹5 小时前
AI量化模型解析黄金3300关口博弈:市场聚焦“非农数据”的GRU-RNN混合架构推演
大数据·人工智能·算法
金融小师妹5 小时前
基于LSTM-GRU混合网络的动态解析:美联储维稳政策与黄金单日跌1.5%的非线性关联
大数据·人工智能·算法
白日梦想家-K5 小时前
题单【模拟与高精度】
开发语言·c++·算法
重生之我是Java开发战士6 小时前
【C语言】内存函数与数据在内存中的存储
c语言·开发语言·算法
roman_日积跬步-终至千里7 小时前
【机器学习】“回归“算法模型的三个评估指标:MAE(衡量预测准确性)、MSE(放大大误差)、R²(说明模型解释能力)
算法·机器学习·回归
小指纹8 小时前
图论-最短路Dijkstra算法
数据结构·c++·算法·深度优先·图论
赴3359 小时前
逻辑回归 银行贷款资格判断案列优化 交叉验证,调整阈值,下采样与过采样方法
算法·机器学习·逻辑回归·下采样·交叉验证·过采样·阈值