LeetCode 1465. 切割后面积最大的蛋糕【贪心,数组,排序】1444

2023-10-28 15:20

本文主要是介绍LeetCode 1465. 切割后面积最大的蛋糕【贪心,数组,排序】1444,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文属于「征服LeetCode」系列文章之一,这一系列正式开始于2021/08/12。由于LeetCode上部分题目有锁,本系列将至少持续到刷完所有无锁题之日为止;由于LeetCode还在不断地创建新题,本系列的终止日期可能是永远。在这一系列刷题文章中,我不仅会讲解多种解题思路及其优化,还会用多种编程语言实现题解,涉及到通用解法时更将归纳总结出相应的算法模板。

为了方便在PC上运行调试、分享代码文件,我还建立了相关的仓库:https://github.com/memcpy0/LeetCode-Conquest。在这一仓库中,你不仅可以看到LeetCode原题链接、题解代码、题解文章链接、同类题目归纳、通用解法总结等,还可以看到原题出现频率和相关企业等重要信息。如果有其他优选题解,还可以一同分享给他人。

由于本系列文章的内容随时可能发生更新变动,欢迎关注和收藏征服LeetCode系列文章目录一文以作备忘。

矩形蛋糕的高度为 h 且宽度为 w,给你两个整数数组 horizontalCuts 和 verticalCuts,其中:

  • horizontalCuts[i] 是从矩形蛋糕顶部到第  i 个水平切口的距离
  • verticalCuts[j] 是从矩形蛋糕的左侧到第 j 个竖直切口的距离

请你按数组 horizontalCuts 和 verticalCuts 中提供的水平和竖直位置切割后,请你找出 面积最大 的那份蛋糕,并返回其 面积 。由于答案可能是一个很大的数字,因此需要将结果  109 + 7 取余 后返回。

示例 1:

输入:h = 5, w = 4, horizontalCuts = [1,2,4], verticalCuts = [1,3]
输出:4 
解释:上图所示的矩阵蛋糕中,红色线表示水平和竖直方向上的切口。切割蛋糕后,绿色的那份蛋糕面积最大。

示例 2:

输入:h = 5, w = 4, horizontalCuts = [3,1], verticalCuts = [1]
输出:6
解释:上图所示的矩阵蛋糕中,红色线表示水平和竖直方向上的切口。切割蛋糕后,绿色和黄色的两份蛋糕面积最大。

示例 3:

输入:h = 5, w = 4, horizontalCuts = [3], verticalCuts = [3]
输出:9

提示:

  • 2 <= h, w <= 10^9
  • 1 <= horizontalCuts.length <= min(h - 1, 10^5)
  • 1 <= verticalCuts.length <= min(w - 1, 10^5)
  • 1 <= horizontalCuts[i] < h
  • 1 <= verticalCuts[i] < w
  • 题目数据保证 horizontalCuts 中的所有元素各不相同
  • 题目数据保证 verticalCuts 中的所有元素各不相同

解法 贪心+排序

为了算出两个相邻切口的距离,把 horizontalCuts \textit{horizontalCuts} horizontalCuts verticalCuts \textit{verticalCuts} verticalCuts 从小到大排序。

排序后,任意蛋糕都可以由两个相邻的水平切口和两个相邻的竖直切口决定(边界也算切口)。例如示例 1,水平切口 2 , 4 2,4 2,4 和竖直切口 1 , 3 1,3 1,3 就决定了面积最大的那份蛋糕。

所以蛋糕的面积可以表示为
( horizontalCuts [ i ] − horizontalCuts [ i − 1 ] ) ⋅ ( verticalCuts [ j ] − verticalCuts [ j − 1 ] ) (\textit{horizontalCuts}[i] - \textit{horizontalCuts}[i-1])\cdot (\textit{verticalCuts}[j] - \textit{verticalCuts}[j-1]) (horizontalCuts[i]horizontalCuts[i1])(verticalCuts[j]verticalCuts[j1])
由于相乘的两项是互相独立的,只需要分别算出 horizontalCuts [ i ] − horizontalCuts [ i − 1 ] \textit{horizontalCuts}[i] - \textit{horizontalCuts}[i-1] horizontalCuts[i]horizontalCuts[i1] 的最大值和 verticalCuts [ j ] − verticalCuts [ j − 1 ] \textit{verticalCuts}[j] - \textit{verticalCuts}[j-1] verticalCuts[j]verticalCuts[j1] 的最大值,就得到了蛋糕面积的最大值。

也可以这样理解,先横着切,那么间隔最大的那两刀,就决定了最大的那条蛋糕,它包含着最终面积最大的那份蛋糕。接下来只需要知道竖着切时,两刀的最大间隔是多少,就知道了蛋糕的最大面积。

class Solution {int get_max_size(int size, vector<int> &cuts) {sort(cuts.begin(), cuts.end());int res = max(cuts[0], size - cuts.back());for (int i = 1; i < cuts.size(); i++) {res = max(res, cuts[i] - cuts[i - 1]);}return res;}
public:int maxArea(int h, int w, vector<int> &horizontalCuts, vector<int> &verticalCuts) {int max_h = get_max_size(h, horizontalCuts);int max_w = get_max_size(w, verticalCuts);return (long long) max_h * max_w % 1'000'000'007;}
};

复杂度分析:

  • 时间复杂度: O ( n log ⁡ n + m log ⁡ m ) \mathcal{O}(n\log n + m\log m) O(nlogn+mlogm) ,其中 n n n horizontalCuts \textit{horizontalCuts} horizontalCuts 的长度, m m m verticalCuts \textit{verticalCuts} verticalCuts 的长度。瓶颈在排序上。
  • 空间复杂度: O ( n ) \mathcal{O}(n) O(n) O ( 1 ) \mathcal{O}(1) O(1)

这篇关于LeetCode 1465. 切割后面积最大的蛋糕【贪心,数组,排序】1444的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Spring排序机制之接口与注解的使用方法

《Spring排序机制之接口与注解的使用方法》本文介绍了Spring中多种排序机制,包括Ordered接口、PriorityOrdered接口、@Order注解和@Priority注解,提供了详细示例... 目录一、Spring 排序的需求场景二、Spring 中的排序机制1、Ordered 接口2、Pri

大数据小内存排序问题如何巧妙解决

《大数据小内存排序问题如何巧妙解决》文章介绍了大数据小内存排序的三种方法:数据库排序、分治法和位图法,数据库排序简单但速度慢,对设备要求高;分治法高效但实现复杂;位图法可读性差,但存储空间受限... 目录三种方法:方法概要数据库排序(http://www.chinasem.cn对数据库设备要求较高)分治法(常

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

Python中lambda排序的六种方法

《Python中lambda排序的六种方法》本文主要介绍了Python中使用lambda函数进行排序的六种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们... 目录1.对单个变量进行排序2. 对多个变量进行排序3. 降序排列4. 单独降序1.对单个变量进行排序

JAVA中整型数组、字符串数组、整型数和字符串 的创建与转换的方法

《JAVA中整型数组、字符串数组、整型数和字符串的创建与转换的方法》本文介绍了Java中字符串、字符数组和整型数组的创建方法,以及它们之间的转换方法,还详细讲解了字符串中的一些常用方法,如index... 目录一、字符串、字符数组和整型数组的创建1、字符串的创建方法1.1 通过引用字符数组来创建字符串1.2

关于Java内存访问重排序的研究

《关于Java内存访问重排序的研究》文章主要介绍了重排序现象及其在多线程编程中的影响,包括内存可见性问题和Java内存模型中对重排序的规则... 目录什么是重排序重排序图解重排序实验as-if-serial语义内存访问重排序与内存可见性内存访问重排序与Java内存模型重排序示意表内存屏障内存屏障示意表Int

如何提高Redis服务器的最大打开文件数限制

《如何提高Redis服务器的最大打开文件数限制》文章讨论了如何提高Redis服务器的最大打开文件数限制,以支持高并发服务,本文给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录如何提高Redis服务器的最大打开文件数限制问题诊断解决步骤1. 修改系统级别的限制2. 为Redis进程特别设置限制

vue如何监听对象或者数组某个属性的变化详解

《vue如何监听对象或者数组某个属性的变化详解》这篇文章主要给大家介绍了关于vue如何监听对象或者数组某个属性的变化,在Vue.js中可以通过watch监听属性变化并动态修改其他属性的值,watch通... 目录前言用watch监听深度监听使用计算属性watch和计算属性的区别在vue 3中使用watchE