求数组给定的容器的蓄水量

题目描述

给定一个非负数的数组,代表一个容器。例如数组[0,1,0,2,1,0,1,3,2,1,2,1],就是 以下图形中黑色的部分。如果用这个容器接水的话,请问可以接多少水?还以这个数组为例,可以接6格水,就是以下图形中蓝色的部分。
要求:实现时间复杂度O(N),额外空间复杂度O(1)的解法。
题目图片

首先看一下简单的算法

解法一:使用辅助数组

原理:假设我要求第i个位置处的水的值,则我应该找到i左边的最大值和i右边的最大值,使用两个数组
L和R来表示。其中L[i]表示i前面的最大值,R[i]表示后面的最大值。然后使用两个最大值中的较小的值去减去i的值,如果i的值小于等于0,则积蓄的水为0,否则就是相减后的值。由于遍历求每个i是从左到右的,每次i左边的最大值只需要一个值来保存。
时间复杂度:O(n),空间复杂度:O(n)

  1. publicstaticint maxWater(int[] arr){
  2. if(arr ==null|| arr.length <=2)return0;
  3. int maxL = arr[0];
  4. int[] maxR =newint[arr.length];
  5. int max =0;
  6. for(int i = arr.length -1; i >=0; i--){
  7. if(arr[i]> max) maxR[i]= max = arr[i];
  8. else maxR[i]= max;
  9. }
  10. max =0;//用来计算总的水量
  11. for(int i =1; i < arr.length -1; i++){
  12. if(arr[i]> maxL) maxL = arr[i];//更新i左边的最大值
  13. max +=Math.max(Math.min(maxL, maxR[i])- arr[i],0);
  14. }
  15. return max;
  16. }
解法二:使用四个变量

5 4 … 3 7
| |
从左右两边向中间遍历,首先设定左右两边的最大值为数组第一个和最后一个数,当左边的值小于右边的值,左边值小于右边值, 则说明左边的值已经可以计算左边的下一个值(这是左边的值已经小于右边的值,因为水量是根据左右最大值的小值确定的,左边的i左边的最大值是确定了,且是小于左右的值,说明计算i时,是使用左边的最大值计算的,因此此时可以计算左边的下一个值),同理当右边最大值小于左边最大值时,可以计算右边的上一个值。
实现时间复杂度O(N),空间复杂度O(1)。
实现:

  1. publicstaticint maxWaterWithConstSpace(int[] arr){
  2. if(arr ==null|| arr.length <=2)return0;
  3. int maxL = arr[0];
  4. int maxR = arr[arr.length -1];
  5. int max =0;//用来计算总的水量
  6. max =0;
  7. int l =1;
  8. int r = arr.length -2;
  9. while(l <= r){
  10. if(maxL <= maxR){
  11. max +=Math.max(0, maxL - arr[l]);
  12. if(arr[l++]> maxL) maxL = arr[l-1];
  13. }else{
  14. max +=Math.max(0, maxR - arr[r]);
  15. if(arr[r--]> maxR) maxR = arr[r +1];
  16. }
  17. }
  18. return max;
  19. }





转载于:https://www.cnblogs.com/ggmfengyangdi/p/5706803.html

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
在C#中,可以使用循环来数组的最大值和最小值。在给定的代码片段中,引用演示了如何找到数组的最大值,而引用演示了如何找到数组的最小值。 要找到数组的最大值,可以使用以下步骤: 1. 首先,定义一个整数变量max,并将其初始化为数组的第一个元素a。 2. 然后,使用一个循环遍历数组中的每个元素。在循环中,从索引0开始,逐个增加索引i,直到i小于数组的长度a.Length。 3. 在循环体中,使用条件语句判断当前元素a[i]是否大于max。如果是,则更新max的值为当前元素a[i]。 4. 循环结束后,max的值将是数组的最大值。 5. 最后,使用Console.WriteLine语句将max的值打印出来。 要找到数组的最小值,可以使用类似的步骤: 1. 首先,定义一个整数变量min,并将其初始化为数组的第一个元素a。 2. 然后,使用一个循环遍历数组中的每个元素。在循环中,从索引0开始,逐个增加索引i,直到i小于数组的长度a.Length。 3. 在循环体中,使用条件语句判断当前元素a[i]是否小于min。如果是,则更新min的值为当前元素a[i]。 4. 循环结束后,min的值将是数组的最小值。 5. 最后,使用Console.WriteLine语句将min的值打印出来。 所以,对于给定数组{10, 9, 6, 4234, 312},最大值是4234,最小值是6。 而对于数组{10, 1, 5, 6123, 34},最大值是6123,最小值是1。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值