一天一大 leet(最小路径和)难度:中等-Day20200723

img

题目:

给定一个包含非负整数的 m x n 网格,请找出一条从左上角到右下角的路径,使得路径上的数字总和为最小。

说明:

每次只能向下或者向右移动一步。

示例:

输入:
[
  [1,3,1],
  [1,5,1],
  [4,2,1]
]
输出: 7
解释: 因为路径 1→3→1→1→1 的总和最小。

抛砖引玉

img

已经刷过的类似题目:

回归之前的逻辑答题就两种方式解决:动态规划、递归

动态规划

思路

  • 记录到达每一格时最小的路径
  • 到记录到 dp[m][n]时就是需要的结果
  • 到达每一格的路径和可能有两种
    • 从上面单元格进入则:dp[i][j] = dp[i][j-1]+grid[i][j]
    • 从上面单元格进入则:dp[i][j] = dp[i-1][j]+grid[i][j]
  • 索引存在-1 则需要注意边界问题:
    • grid 长 0,返回 0
    • dp[0][0] = grid[0][0]
    • 遍历从 i=1,j=1 开始
    • dp[i][0],dp[0][j],使用当前行或者列累加做默认值
123
12
2->dp[i][j]
3
/**
 * @param {number[][]} grid
 * @return {number}
 */
var minPathSum = function (grid) {
  let m = grid.length,
    n = grid[0] ? grid[0].length : 0
  if (m === 0 || n === 0) return 0
  let dp = Array.from({ length: m }, () => new Array(n).fill(Number.MAX_VALUE))

  // 起点
  dp[0][0] = grid[0][0]

  // 补齐首行路径和
  for (let i = 1; i < m; i++) {
    dp[i][0] = dp[i - 1][0] + grid[i][0]
  }

  // 补齐首列路径和
  for (let j = 1; j < n; j++) {
    dp[0][j] = dp[0][j - 1] + grid[0][j]
  }

  for (let i = 1; i < m; i++) {
    for (let j = 1; j < n; j++) {
      // 从路径和较小的地方进入
      let itemMin = Math.min(dp[i - 1][j], dp[i][j - 1])
      dp[i][j] = Math.min(itemMin + grid[i][j], dp[i][j])
    }
  }
  return dp[m - 1][n - 1]
}

降维

  • dp 降维不记录到每个单元格的最小路径
  • dp 记录到达每行的最小路径
d[0]d[i]
12
23
3grid[i][j]
  • 行路径放置发哦 dp[0]
  • 列路径放置到 dp[j]
/**
 * @param {number[][]} grid
 * @return {number}
 */
var minPathSum = function (grid) {
  let m = grid.length,
    n = grid[0] ? grid[0].length : 0
  if (m === 0 || n === 0) return 0

  let dp = new Array(n).fill(0)

  dp[0] = grid[0][0]

  // 补齐首列路径和
  for (let i = 1; i < n; i++) {
    dp[i] = dp[i - 1] + grid[0][i]
  }

  for (let i = 1; i < m; i++) {
    // 开始每列循环时限初始化本列起点路径和,到达本列前的路径和
    dp[0] = dp[0] + grid[i][0]
    for (let j = 1; j < n; j++) {
      dp[j] = Math.min(dp[j - 1], dp[j]) + grid[i][j]
    }
  }
  return dp[n - 1]
}

递归

  • 计算路径的逻辑还是和上面一样,入口+当前单元格中的值
  • 递归参数,行列(i,j)
  • 递归边界条件:
    • i 和 j 任意指针越界说明指针并为指向单元格丢弃该值,使用最大值在取 min 时会自行丢弃
    • dp 存储过此单元格路径和直接返回存储在 dp 中的值
/**
 * @param {number[][]} grid
 * @return {number}
 */
var minPathSum = function (grid) {
  let m = grid.length,
    n = grid[0] ? grid[0].length : 0
  if (m === 0 || n === 0) return 0

  let dp = Array.from({ length: m }, () => Array(n).fill(0))

  // 起点
  dp[0][0] = grid[0][0]

  function get_sum(i, j) {
    if (i < 0 || j < 0) return Number.MAX_VALUE
    if (dp[i][j]) return dp[i][j]
    let itemMin = grid[i][j] + Math.min(get_sum(i - 1, j), get_sum(i, j - 1))
    dp[i][j] = itemMin
    return itemMin
  }

  return get_sum(m - 1, n - 1)
}

超时

可以看到有一个用例没有通过,可以借助降维的思路,对存储的 dp 降维

/**
 * @param {number[][]} grid
 * @return {number}
 */
var minPathSum = function (grid) {
  let m = grid.length,
    n = grid[0] ? grid[0].length : 0
  if (m === 0 || n === 0) return 0

  let dp = new Map()

  // 起点
  dp.set('0-0', grid[0][0])

  function get_sum(i, j) {
    if (i < 0 || j < 0) return Number.MAX_VALUE
    if (dp.has(i + '-' + j)) return dp.get(i + '-' + j)
    let itemMin = grid[i][j] + Math.min(get_sum(i - 1, j), get_sum(i, j - 1))
    dp.set(i + '-' + j, itemMin)
    return itemMin
  }

  return get_sum(m - 1, n - 1)
}

超时

通过,完结撒花

(((┏(; ̄ ▽  ̄)┛ 装完逼就跑

博客: 小书童博客

每天的每日一天,写的题解会同步更新到公众号一天一大 lee 栏目

公号: 坑人的小书童

坑人的小书童

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
探险家小扣的行动轨迹,都将保存在记录仪中。expeditions[i] 表示小扣第 i 次探险记录,用一个字符串数组表示。其中的每个「营地」由大小写字母组成,通过子串 -> 连接。例:"Leet->code->Campsite",表示到访了 "Leet"、"code"、"Campsite" 三个营地。expeditions[0] 包含了初始小扣已知的所有营地;对于之后的第 i 次探险(即 expeditions[i] 且 i > 0),如果记录中包含了之前均没出现的营地,则表示小扣 新发现 的营地。 请你找出小扣发现新营地最多且索引最小的那次探险,并返回对应的记录索引。如果所有探险记录都没有发现新的营地,返回 -1。注意: 大小写不同的营地视为不同的营地; 营地的名称长度均大于 0。用python实现。给你几个例子:示例 1: 输入:expeditions = ["leet->code","leet->code->Campsite->Leet","leet->code->leet->courier"] 输出:1 解释: 初始已知的所有营地为 "leet" 和 "code" 第 1 次,到访了 "leet"、"code"、"Campsite"、"Leet",新发现营地 2 处:"Campsite"、"Leet" 第 2 次,到访了 "leet"、"code"、"courier",新发现营地 1 处:"courier" 第 1 次探险发现的新营地数量最多,因此返回 1。示例 2: 输入:expeditions = ["Alice->Dex","","Dex"] 输出:-1 解释: 初始已知的所有营地为 "Alice" 和 "Dex" 第 1 次,未到访任何营地; 第 2 次,到访了 "Dex",未新发现营地; 因为两次探险均未发现新的营地,返回 -1
04-23
我的理解是,你想让我编写一段 Python 代码,实现一个算法,用于在给定的探险记录中找到发现新营地最多且索引最小的探险记录。 那么首先,我需要解析题目中的输入输出要。 输入要: - expeditions:一个字符串数组,表示小扣第一次到访的所有营地,和之后的每次探险记录(一个记录就是一个字符串,其中营地通过子串 -> 连接)。 输出要: - 如果所有的探险记录都没有发现新的营地,则返回 -1; - 否则,返回发现新营地最多且索引最小的探险记录的索引。注:探险记录的索引从 0 开始计数。 接下来,我可以考虑如何编写代码来解决这个问题。 我的思路是,首先用一个集合记录所有已知的营地名称,然后遍历每个探险记录,依次将其中的营地名称加入到集合中,并判断是否有新的营地被发现。 如果某个探险记录中发现了新的营地,那么我就记录下发现了多少个新营地,以及这个探险记录的索引。最后,遍历所有的探险记录,找到发现新营地最多且索引最小的记录。 代码如下: ```python def find_new_camp(expeditions): known_camps = set() # 已知的营地集合 max_new_camps = 0 # 最多的新营地数量 max_new_camps_index = -1 # 最多新营地数量的探险记录的索引 for i in range(len(expeditions)): sub_camps = expeditions[i].split("->") # 提取探险记录中所有的营地名称 num_new_camps = 0 # 记录这个探险记录发现的新营地数量 for camp in sub_camps: if camp not in known_camps: # 如果这个营地是新的,就记录一下 known_camps.add(camp) num_new_camps += 1 if num_new_camps > max_new_camps: # 如果发现了更多的新营地,就更新最大值和索引 max_new_camps = num_new_camps max_new_camps_index = i return max_new_camps_index if max_new_camps > 0 else -1 ``` 我使用了两个变量 `max_new_camps` 和 `max_new_camps_index` 来记录已经遍历过的探险记录中,发现了最多的新营地数量,以及具有最多新营地数量的探险记录的索引。然后我遍历所有的探险记录,统计其中发现的新营地数量,如果大于当前的最大值,就更新 `max_new_camps` 和 `max_new_camps_index` 变量。最后,如果发现了新的营地,就返回 `max_new_camps_index`。如果所有的探险记录都没有发现新营地,就返回 -1。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值