当前位置:   article > 正文

每日OJ题_算法_双指针④_力扣11. 盛最多水的容器

每日OJ题_算法_双指针④_力扣11. 盛最多水的容器

目录

力扣11. 盛最多水的容器

解析代码


力扣11. 盛最多水的容器

11. 盛最多水的容器 - 力扣(LeetCode)

难度 中等

给定一个长度为 n 的整数数组 height 。有 n 条垂线,第 i 条线的两个端点是 (i, 0) 和 (i, height[i]) 。

找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。

返回容器可以储存的最大水量。

说明:你不能倾斜容器。

示例 1:

输入:[1,8,6,2,5,4,8,3,7]
输出:49 
解释:图中垂直线代表输入数组 [1,8,6,2,5,4,8,3,7]。在此情况下,容器能够容纳水(表示为蓝色部分)的最大值为 49。

示例 2:

输入:height = [1,1]
输出:1

提示:

  • n == height.length
  • 2 <= n <= 10^5
  • 0 <= height[i] <= 10^4
  1. class Solution {
  2. public:
  3. int maxArea(vector<int>& height) {
  4. }
  5. };

解析代码

首先想到的是两层循环的暴力解法,时间复杂度是O(N^2),这里采用双指针(对撞指针)的思想优化到O(N):

设两个指针 left , right 分别指向容器的左右两个端点,此时容器的容积 :
v = (right - left) * min(height [ right ], height [ left ] )
容器的左边界为 height [ left ]  ,右边界为 height [ right ] 。
为了方便叙述,假设「左边边界」小于「右边边界」。

  • 容器的宽度一定变小。
  • 由于左边界较小,决定了水的高度。如果改变左边界,新的水面高度不确定,但是一定不会超过右边的柱子高度,因此容器的容积可能会增大。
  • 如果改变右边界,无论右边界移动到哪里,新的水面的高度一定不会超过左边界,也就是不会超过现在的水面高度,但是由于容器的宽度减小,因此容器的容积一定会变小。

由此可见,左边界和其余边界的组合情况都可以舍去。所以可以left++跳过这个边界,继续去判断下一个左右边界。

不断重复上述过程,每次都可以舍去大量不必要的枚举过程,直到left与right相遇。期间产生的所有的容积里面的最大值,就是最终答案。

代码:

  1. class Solution {
  2. public:
  3. int maxArea(vector<int>& height) {
  4. int left = 0, right = height.size() - 1, ret = 0;
  5. while(left < right)
  6. {
  7. int v = (right - left) * min(height[left], height[right]);
  8. ret = max(v, ret);
  9. if(height[left] < height[right]) // 哪个小哪个就往中间移动
  10. {
  11. ++left;
  12. }
  13. else
  14. {
  15. --right;
  16. }
  17. }
  18. return ret;
  19. }
  20. };
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/羊村懒王/article/detail/194531
推荐阅读
相关标签
  

闽ICP备14008679号