接雨水

接雨水

给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。[^1]

在这里插入图片描述

问题分析

 根据木桶原理,空间最多接雨水的量取决于当前空间相对最短的高度,题目中的则是两个柱子中的相对较短的一端决定。而在题中,又存在这样一个问题,父空间包含子空间的情况,为此,不能简单将各个柱子之间构成的空间简单相加,我们还需确立空间中的父子空间的关系。怎么去确定父子关系?在一个区间里面的柱子如果存在比区间端口处的任意一个端口高的情况下,这个区间端口构成的空间就不是这个区间的父空间,这时,在这个区间一定存在两个及以上独立的空间。为此我们再将这个空间进一步划分成几个空间进行计算,这是一个思路。
另一个思路,我们可以模仿倒水的过程。以高度为主要目标。比如高度为一时,各个柱子是否存在,以及是否存在空间(及中间为0两端为1的情况),空间的大小,转换成bitmaps进行计算。

怎么用计算机的思维去思考这个问题呢??
1)可以将题中的柱子用数组表示heighti(n为柱子个数减一),
柱子可以围成的空间表示为area[i]为(0-i)空间可以盛水的数量。
2)为此,我们可以利用动态规划和回溯的思想进行分析,我们现在的目的就是为了通过area[i]找到area[i+1]的盛水空间。怎么找到这个关系呢?首先我们需要知道的是(0-i)空间上的最高柱子的坐标,记为cur_max_pos,i+1上的柱子的高度,现在我们要做的就是找(cur_max_pos,i+1)空间关系,那么我们可以以i点为起始点,往回找,当找到比i+1处高度高的说明i+1处的最大空间以找到,当i+1搜索到cur_max_pos仍然未找到比i+1高的,则,i+1处在area[i]处的影响范围触及到的最大范围处是在cur_max_pos处。那么加入i+1之后area的值就变成了area[i]中未受i+1影响的值加上i+1受影响的值。

代码实现

var trap = function(height) {
    var cur_pos=0;
    var cur_max_pos=0;
    var area=[];
    area[0]=0;

    if(height.length==0){
        return 0;
    }
    for(var i=0;i<height.length;i++){

        var tmp_area=0;
        var flag=0;
        for(var j=cur_pos;j>=cur_max_pos;j--){
            
            if(height[i]>height[j]){
                tmp_area+=height[j];
                continue;
            }

            area[i]=area[j]+height[i]*(i-j-1)-tmp_area;
            flag=1;
            break;
        }

        if(flag==0){
            area[i]=area[cur_max_pos]+height[cur_max_pos]*(i-j-1)-tmp_area;
        }

        cur_pos=i;
        if(height[cur_max_pos]<height[cur_pos]){
            cur_max_pos=cur_pos;
        }

        if(area[i]<0){
        	area[i]=0;
        }
    }

    if(area[height.length-1]<0){
        return 0;
    }else{
        return area[height.length-1];
    }
};

[^1]https://leetcode-cn.com/problems/trapping-rain-water/submissions/

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值