想要精通算法和SQL的成长之路 - 填充书架

这篇具有很好参考价值的文章主要介绍了想要精通算法和SQL的成长之路 - 填充书架。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

前言

想要精通算法和SQL的成长之路 - 系列导航

一. 填充书架

原题链接
想要精通算法和SQL的成长之路 - 填充书架,精通算法和SQL之路,算法
想要精通算法和SQL的成长之路 - 填充书架,精通算法和SQL之路,算法
想要精通算法和SQL的成长之路 - 填充书架,精通算法和SQL之路,算法

题目中有一个值得注意的点就是:

  • 需要按照书本顺序摆放。
  • 每一层当中,只要厚度不够了,当前层最高的那一本书籍就视为本层的高度。

那么我们假设dp[i]: 代表从 book[0] 摆到 book[i] 的时书架的最小高度。

  • 假设最后一层的第一本书的下标是 j,那么之前所有书本摆放的最小高度就是 dp[j-1]
  • 我们再计算出,下标在[j,i](最后一层)的书本中,高度最高的那一本书(同时满足厚度不超过shelfWidth),高度为maxHeight
  • 那么当前的最小总高度是 res = Max(dp[i-1]+maxHeight,res)。即之前的总高度+最后一层的最高高度。

我们递归,从后往前递归。入参为遍历的书本下标。

  1. 终止条件:下标 <0。代表没有书本了,停止递归。
  2. 递归做的事情:循环[0,i]之间的所有元素,从后往前把书本放入最后一层,一旦厚度超出,终止遍历。否则,计算当前层的最高高度以及最小总高。
public class Test1105 {
    public int[][] books;
    public int shelfWidth;

    public int minHeightShelves(int[][] books, int shelfWidth) {
        this.books = books;
        this.shelfWidth = shelfWidth;
        return dfs(books.length - 1);
    }

    public int dfs(int i) {
        // 终止条件
        if (i < 0) {
            return 0;
        }
        int res = Integer.MAX_VALUE, maxHeight = 0, width = shelfWidth;
        for (int j = i; j >= 0; j--) {
            // 从后往前放书本
            width -= books[j][0];
            // 厚度不能超过 shelfWidth ,超过就代表放不下了
            if (width < 0) {
                break;
            }
            // 当前层最高高度
            maxHeight = Math.max(maxHeight, books[j][1]);
            // 更新总最低书架高度 = 上层最小总高度 + 当前层最高高度
            res = Math.min(res, dfs(j - 1) + maxHeight);
        }
        return res;
    }
}

这个解答其实对于用例比较多的情况,是会超时的。

1.1 优化

我们来看下上面代码的不好的点:

  • 每次dfs的时候,循环的范围是:[0,j]
  • 循环内部又每次调用了dfs递归,即dfs[j-1]

整个递归函数,只用到了一个索引的参数,我们可以发现,索引为1,2,3…的递归,被重复调用了非常多次。以当前索引为3为例:

  • 第一次递归范围:[0,3]。
  • 第二次递归范围:[0,2]。
  • 第三次递归范围:[0,1]。

那么我们可以用一个全局的变量去记录每次dfs返回的结果即可:文章来源地址https://www.toymoban.com/news/detail-733353.html

public class Test1105 {
    public int[][] books;
    public int shelfWidth;
    // 缓存dfs的结果
    public int[] dfsCache;

    public int minHeightShelves(int[][] books, int shelfWidth) {
        this.books = books;
        this.shelfWidth = shelfWidth;
        // 初始化
        dfsCache = new int[books.length];
        // 给个初始值,-1代表没有被执行过,即没有缓存
        Arrays.fill(dfsCache, -1);
        return dfs(books.length - 1);
    }

    public int dfs(int i) {
        // 终止条件
        if (i < 0) {
            return 0;
        }
        // 如果是-1,代表这层值没有执行过,往下走。否则,说明有缓存了,直接返回
        if (dfsCache[i] != -1) {
            return dfsCache[i];
        }
        int res = Integer.MAX_VALUE, maxHeight = 0, width = shelfWidth;
        for (int j = i; j >= 0; j--) {
            // 从后往前放书本
            width -= books[j][0];
            // 厚度不能超过 shelfWidth ,超过就代表放不下了
            if (width < 0) {
                break;
            }
            // 当前层最高高度
            maxHeight = Math.max(maxHeight, books[j][1]);
            // 更新总最低书架高度 = 上层最小总高度 + 当前层最高高度
            res = Math.min(res, dfs(j - 1) + maxHeight);
        }
        // 缓存下当前结果
        dfsCache[i] = res;
        return dfsCache[i];
    }
}

到了这里,关于想要精通算法和SQL的成长之路 - 填充书架的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 想要精通算法和SQL的成长之路 - 滑动窗口和大小根堆

    想要精通算法和SQL的成长之路 - 系列导航 先来说下大小根堆是什么: 大根堆:栈顶元素最大(上图左侧部分),栈底至栈顶元素值递增。 小根堆:栈顶元素最小(上图右侧部分),栈底至栈顶元素值递减。 在 Java 当中,可以用什么来表示大小根堆? 小根堆: 大根堆: 大小

    2024年02月07日
    浏览(43)
  • 想要精通算法和SQL的成长之路 - 二叉树的序列化和反序列化问题

    想要精通算法和SQL的成长之路 - 系列导航 二叉树的层序遍历 像这种从上至下并且按层打印的,可以称之为 二叉树的广度优先搜索( BFS ) 。而这类算法往往借助 队列的一个先入先出特性 来实现。 那么有这么几个步骤: 1.特殊处理还有初始化动作。 2. BFS 循环: 最终完整代

    2024年02月07日
    浏览(49)
  • 【算法】Filling Bookcase Shelves 填充书架

    给定一个数组 books ,其中 b o o k s [ i ] = [ t h i c k n e s s i , h e i g h t i ] books[i] = [thicknessi, heighti] b oo k s [ i ] = [ t hi c kn ess i , h e i g h t i ] 表示第 i 本书的厚度和高度。你也会得到一个整数 shelfWidth 。 按顺序 将这些书摆放到总宽度为 shelfWidth 的书架上。 先选几本书放在书架上

    2024年02月08日
    浏览(77)
  • 斗破苍穹算法——萧炎的成长之路(二)

    「作者主页」 :雪碧有白泡泡 「个人网站」 :雪碧的个人网站 「推荐专栏」 : ★ java一站式服务 ★ ★ 前端炫酷代码分享 ★ ★ uniapp-从构建到提升 ★ ★ 从0到英雄,vue成神之路 ★ ★ 解决算法,一个专栏就够了 ★ ★ 架构咱们从0说 ★ ★ 数据流通的精妙之道★ 萧炎是一

    2024年02月14日
    浏览(37)
  • 斗破苍穹算法版—萧炎的成长之路

    「作者主页」 :雪碧有白泡泡 「个人网站」 :雪碧的个人网站 「推荐专栏」 : ★ java一站式服务 ★ ★ 前端炫酷代码分享 ★ ★ uniapp-从构建到提升 ★ ★ 从0到英雄,vue成神之路 ★ ★ 解决算法,一个专栏就够了 ★ ★ 架构咱们从0说 ★ ★ 数据流通的精妙之道★ 萧炎是一

    2024年02月16日
    浏览(36)
  • 斗破苍穹算法版—萧炎的成长之路(一)

    「作者主页」 :雪碧有白泡泡 「个人网站」 :雪碧的个人网站 「推荐专栏」 : ★ java一站式服务 ★ ★ 前端炫酷代码分享 ★ ★ uniapp-从构建到提升 ★ ★ 从0到英雄,vue成神之路 ★ ★ 解决算法,一个专栏就够了 ★ ★ 架构咱们从0说 ★ ★ 数据流通的精妙之道★ 萧炎是一

    2024年02月16日
    浏览(41)
  • 我的“测试开发”成长之路

    我相信,有很多测试人员会不断问自己,自己到底要不要坚持做测试,测试的职业发展到底怎么样?如果你还在迷茫,在到处找各种大牛问类似的问题,我希望这篇文章,你看完能够结束你的这个烦恼,给你更多的指明方向,当然也有更多的压力。        这个问题,就像

    2024年02月02日
    浏览(68)
  • Dockerfile成长之路

    随着业务架构的整改,针对非容器化业务全部进行容器化改造,这就设计到了java写的业务代码构建业务镜像,并通过k8s发版,因此,就得学习如何使用dockerfile构建后端业务镜像,可能不止构建后端代码镜像,例如前端写的代码也有可能构建为镜像。还有可能就是要在原有镜像基础上进

    2024年01月24日
    浏览(49)
  • Android程序员成长之路

    应该热爱学习Android知识 具备基本的自学能力和解决问题的能力 具备实践能力 Java(基本) C/C++(进阶) Kotlin(基本) Python(可选) 飞书学习路线图 学习路线图正在完善中... 当然读者也可以提出宝贵建议。 我将会按照 Android学习路线图 发布博客文章。 因本人才疏学浅,博客文章中难

    2024年02月09日
    浏览(45)
  • 一个女程序员的成长之路

    2013年大学毕业了,带着迷茫与好玩,我还年轻的心态,开始在郑州寻觅工作机会,最后很荣幸的在一家小公司入职了,工作的内容是给种植大棚的用户打电话,推销农药。每天就是在网上各种农业平台上面找号码,打电话, 一天拨打电话的在四十个左右,却累的都说不出话

    2024年02月14日
    浏览(56)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包