00 问题

00-1 描述

对给定高度为n的一个整数三角形,找出从顶部到底部的最小路径和。每个整数只能向下移动到与之相邻的整数。

找到一个一样的力扣题:120. 三角形最小路径和 - 力扣(LeetCode) (leetcode-cn.com)

示例1:
输入:triangle = [[2],[3,4],[6,5,7],[4,1,8,3]]
输出:11
解释:如下面简图所示:
  2
 3 4
6 5 7
4 1 8 3
自顶向下的最小路径和为 11(即,2 + 3 + 5 + 1 = 11)。
   
示例2:
输入:triangle = [[-10]]
输出:-10

00-2 提示:

1 <= triangle.length <= 200
triangle[0].length == 1
triangle[i].length == triangle[i - 1].length + 1
-104 <= triangle[i][j] <= 104

01 思路

想用动态规划写出来,重点在于状态转移方程。

将等腰三角形抽象为等腰直角三角形,如下

  0 1 2 3
0 2
1 3 4
2 6 5 7
3 8 3 9 2

加上下标化的序列,我们就可以用二维数组dp来考虑。dp是用来存储到i,j位置后用到的最短路径长度,比如dp[2] [2]=2+4+7=13

定义一个起点:

dp[0][0] = a[0][0];

三种情况:

  1. 三角形左路,在直角图里就是第一列,满足:

    dp[i][0]=dp[i-1][0];
  2. 三角形右路,在直角图里是对角线,满足:

    dp[i][i]=dp[i-1][i-1]+a[i][i]
  3. 普通位置

    dp[i][j]=min(dp[i-1][j-1],dp[i-1][j])+a[i][j];

这样程序就很好写了。就是往dp数组里填数就行,最后筛出最后一行的最小值就行。

02 代码

class Solution {
public:
   int minimumTotal(vector<vector<int>>& triangle) {
       int len = triangle.size();
       int dp[200][200]={0};
       dp[0][0]=triangle[0][0];
       for(int i=1;i<len;i++){
           dp[i][0] = dp[i-1][0]+triangle[i][0];
      }
       for(int i=1;i<len;i++){
           dp[i][i] = triangle[i][i]+dp[i-1][i-1];
      }
       for(int i=2;i<len;i++){
           for(int j=1;j<i;j++){
               dp[i][j] = triangle[i][j]+min(dp[i-1][j], dp[i-1][j-1]);
          }
      }
       //填充dp
       //下面筛选路径最短
       int ans = dp[len-1][0];
       for(int j = 1;j < len;j++){
           if(dp[len-1][j]<ans){
               ans = dp[len-1][j];
          }
      }
       return ans;
  }
};

03 升级版--记录路径

03-1 思路

如果要记下路径,需要再来一个二维数组pre来记录坐标为i,j的点的前一个节点。那么如何记录呢?我们看一下:

  • (i,i)的前一个节点就是(i-1,i-1);

  • (i,j)的前一个节点是(i-1,j)或者(i-1,j-1);

  • (i,0)的前一个节点是(i-1,0)。

容易从这些情况总结出,上一节点一定为i-1,只需记录j的值即可。故我们在pre二维数组里记录的就是当前节点的前一节点的j值。

记录之后,我们还需要输出这个最小路径。怎么输出呢?

我们在前一问题的基础上得到最后行的最小值的列值后,将它返回主控函数,并用它作为参数给路径输出函数Disppath。

输出方法为:

  • 对于当前节点,入栈,查它的pre数组值,更新,继续该操作,直到完成。

    更新操作为:

     path.push_back(a[i][k]);
    k=pre[i][k];
  • 逐个出栈。

03-2 代码

//三角形最小路径
#include<stdio.h>
#include<vector>
#include<string.h>
using namespace std;
#define MAXN 100
int a[MAXN][MAXN];//三角形
int n;//高度
int ans = 0;//应求的路径长度
int dp[MAXN][MAXN];
int pre[MAXN][MAXN];//前驱结点
//根据状态转移方程,只记录列号即可
int Search(){
   int i,j;
   dp[0][0] = a[0][0];
   for(i=1;i<n;i++){
       dp[i][0]=dp[i-1][0]+a[i][0];
       pre[i][0]=i-1;
  }
   for(i=1;i<n;i++){
       dp[i][i]=dp[i-1][i-1]+a[i][i];
       pre[i][i]=i-1;
  }
   for(i=2;i<n;i++){
       for(j=1;j<i;j++){
           if(dp[i-1][j-1]<dp[i-1][j]){
               dp[i][j]=dp[i-1][j-1]+a[i][j];
               pre[i][j]=j-1;
          }
           else{
               dp[i][j]=dp[i-1][j]+a[i][j];
               pre[i][j]=j;          
          }
      }
  }
   ans = dp[n-1][0];
   int k=0;
   for ( j = 1; j < n; j++)
  {
       if(ans>dp[n-1][j]){
           ans = dp[n-1][j];
           k=j;
      }
  }
   return k;
}
void Disppath(int k){
   int i=n-1;
   vector<int>path;//路径节点
   while(i>=0){
       path.push_back(a[i][k]);
       k=pre[i][k];
       i--;
  }
   vector<int>::reverse_iterator it;
   for(it=path.rbegin();it!=path.rend();++it){
       printf("%d ",*it);
  }
   printf("\n");
}

int main(){
   int k;//k行k列的三角形
   memset(pre, 0, sizeof(pre));
   memset(dp, 0, sizeof(dp));
   scanf("%d",&n);
   for(int i=0;i<n;i++){
       for(int j=0;j<=i;j++){
           scanf("%d",&a[i][j]);
      }
  }
   k=Search();
   Disppath(k);
   printf("%d\n",ans);
   return 0;
}

算法学习->求解三角形最小路径及其值的更多相关文章

  1. 算法学习->求解三角形最小路径

    00 问题 00-1 描述 对给定高度为n的一个整数三角形,找出从顶部到底部的最小路径和.每个整数只能向下移动到与之相邻的整数. 找到一个一样的力扣题:120. 三角形最小路径和 - 力扣(LeetC ...

  2. 算法学习记录-图——最小路径之Floyd算法

    floyd算法: 解决任意两点间的最短路径的一种算法,可以正确处理有向图或负权的最短路径问题,同时也被用于计算有向图的传递闭包. 设为从到的只以集合中的节点为中间节点的最短路径的长度. 若最短路径经过 ...

  3. leetcode 120. 三角形最小路径和 及 53. 最大子序和

    三角形最小路径和 问题描述 给定一个三角形,找出自顶向下的最小路径和.每一步只能移动到下一行中相邻的结点上. 例如,给定三角形: [ [2], [3,4], [6,5,7], [4,1,8,3] ] ...

  4. [leetcode-120] 三角形最小路径和

    三角形最小路径和 (1过) 给定一个三角形,找出自顶向下的最小路径和.每一步只能移动到下一行中相邻的结点上. 例如,给定三角形: [ [2], [3,4], [6,5,7], [4,1,8,3] ] ...

  5. 领扣-120 三角形最小路径和 Triangle MD

    三角形最小路径和 Triangle 数组 动态规划 问题 给定一个三角形,找出自顶向下的最小路径和.每一步只能移动到下一行中相邻的结点上. 例如,给定三角形: [2], [3,4], [6,5,7], ...

  6. Java实现 LeetCode 120 三角形最小路径和

    120. 三角形最小路径和 给定一个三角形,找出自顶向下的最小路径和.每一步只能移动到下一行中相邻的结点上. 例如,给定三角形: [ [2], [3,4], [6,5,7], [4,1,8,3] ] ...

  7. 1. 线性DP 120. 三角形最小路径和

    经典问题: 120. 三角形最小路径和  https://leetcode-cn.com/problems/triangle/ func minimumTotal(triangle [][]int) ...

  8. [算法]LeetCode 120:三角形最小路径和

    题目描述: 给定一个三角形,找出自顶向下的最小路径和.每一步只能移动到下一行中相邻的结点上. 例如,给定三角形: [ [2], [3,4], [6,5,7], [4,1,8,3]]自顶向下的最小路径和 ...

  9. LeetCode(120):三角形最小路径和

    Medium! 题目描述: 给定一个三角形,找出自顶向下的最小路径和.每一步只能移动到下一行中相邻的结点上. 例如,给定三角形: [ [2], [3,4], [6,5,7], [4,1,8,3] ] ...

随机推荐

  1. 『GoLang』fmt包的使用

    目录 1. fmt 包初识 2. 格式化 verb 应用 2.1 通用 2.2 布尔值 2.3 整数 2.4 浮点数与复数 2.5 字符串和 []byte 2.6 指针 2.7 其他 flag 2.8 ...

  2. 鸿蒙内核源码分析(消息队列篇) | 进程间如何异步传递大数据 | 百篇博客分析OpenHarmony源码 | v33.02

    百篇博客系列篇.本篇为: v33.xx 鸿蒙内核源码分析(消息队列篇) | 进程间如何异步传递大数据 | 51.c.h .o 进程通讯相关篇为: v26.xx 鸿蒙内核源码分析(自旋锁篇) | 自旋锁 ...

  3. AT2305-[AGC010D]Decrementing【博弈论】

    正题 题目链接:https://www.luogu.com.cn/problem/AT2305 题目大意 \(n\)个数字两个人进行博弈,每个人的操作为 选择一个大于1的数字减一 之后所有数字除以所有 ...

  4. Sentry 监控 - Distributed Tracing 分布式跟踪

    系列 1 分钟快速使用 Docker 上手最新版 Sentry-CLI - 创建版本 快速使用 Docker 上手 Sentry-CLI - 30 秒上手 Source Maps Sentry For ...

  5. farOs 介绍

    nGame nGame 一款文字游戏服务端框架;用于快速构建:自由探索.武侠.修真.模拟,回合制,剧本杀.动态语言小说.等服务器 如果你有期望实现的功能请加Q群 ngame计划 完善框架功能 farO ...

  6. react-native移动端设置android闪屏页

    前言 因为app启动时会白屏一段时间,导致让人用起来非常的不舒服,后来了解一下知道这叫做闪屏 于是着手解决这个白屏的问题,换个颜色?不行,不如用一张好看的图片来替换,这样才让人看起来更加舒服. 那么该 ...

  7. Google Chrome打开权限设置开关(摄像头,录音等)

    在搜索框输入以下字符 chrome://flags/#unsafely-treat-insecure-origin-as-secure

  8. css新增属性之边框

    css3新增属性 边框属性 背景属性 文字属性 颜色属性 边框属性 属性 说明 border-radius 设置边框圆角 border-image 设置图像边框 border-shadow 设置边框阴 ...

  9. iframe、SameSite与CEF

    iframe.SameSite与CEF 背景 本人使用CEF(或是Chrome)来加载开发的前端页面,其中使用iframe嵌入了第三方页面,在第三方页面中需要发送cookie到后端,然而加载会报错,第 ...

  10. 6岁!是时候重新认识下Serverless了

    一.背景 Serverless 概念从2012年开始提出,真正推出相关云产品是2014年AWS推出Lambda.如果我们将 Serverless 比作一个婴儿,那么它已经6岁了. 虽然业界对Serve ...