力扣134. 加油站

题目链接

在一条环路上有 N 个加油站,其中第 i 个加油站有汽油 gas[i] 升。
你有一辆油箱容量无限的的汽车,从第 i 个加油站开往第 i+1 个加油站需要消耗汽油 cost[i] 升。你从其中的一个加油站出发,开始时油箱为空。
如果你可以绕环路行驶一周,则返回出发时加油站的编号,否则返回 -1。
说明:
如果题目有解,该答案即为唯一答案。
输入数组均为非空数组,且长度相同。
输入数组中的元素均为非负数。

输入:
gas = [1,2,3,4,5]
cost = [3,4,5,1,2]
输出: 3
解释:
从 3 号加油站(索引为 3 处)出发,可获得 4 升汽油。此时油箱有 = 0 + 4 = 4 升汽油
开往 4 号加油站,此时油箱有 4 - 1 + 5 = 8 升汽油
开往 0 号加油站,此时油箱有 8 - 2 + 1 = 7 升汽油
开往 1 号加油站,此时油箱有 7 - 3 + 2 = 6 升汽油
开往 2 号加油站,此时油箱有 6 - 4 + 3 = 5 升汽油
开往 3 号加油站,你需要消耗 5 升汽油,正好足够你返回到 3 号加油站。
因此,3 可为起始索引。


分析:
最暴力的想法,每个起点枚举一次,然后向后走n步,能走就是答案,所有的都走不了就输出-1。这样做复杂度为O(n^2)
我们可以从中找到某些性质。
在这里插入图片描述
这是上面的样例,我们假设从5开始走,经历如下步骤(5是拥有的油量,因为正好有序,所以也就当序号了):

  1. 从5走到1,有5升油,花了2生,到1的位置剩下3升
  2. 从1走到2,多了1升,花了3生,3 + 1 - 3 = 1升
  3. 从2走不到3了,1 + 2只有3升,不足以走到3

此时我们思考,那5开始走不到,1和2可以吗?
不可以,原因如下:
既然5可以走到1和2,那么以1为例,到1时一定是剩下油的,至少也要是0
之后从1出发,加上之前剩下的油都没走到3,那么把前面的油去掉就更不可能走到了
所以5出发到2失败了,那么下一次直接从3开始即可,这样复杂度就降低为O(n)了

class Solution {
    public int canCompleteCircuit(int[] gas, int[] cost) {
        int n = gas.length;
        for (int i = 0, j; i < n;) { // 从i开始走
            int surplus = 0; // 一开始剩余油量为0
            for (j = 0; j < n; j ++ ) {
                int k = (i + j) % n; // 当前走到的点
                surplus += gas[k] - cost[k]; 
                if (surplus < 0) break;
            }
            if (surplus >= 0) return i;
            i = i + j + 1; // 如果没走完的话中间的其实也不用走了
        }

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值