leetcode134-加油站

原题

在一条环路上有 N 个加油站,其中第 i 个加油站有汽油 gas[i] 升。

你有一辆油箱容量无限的的汽车,从第 i 个加油站开往第 i+1 个加油站需要消耗汽油 cost[i] 升。你从其中的一个加油站出发,开始时油箱为空。

如果你可以绕环路行驶一周,则返回出发时加油站的编号,否则返回 -1。

说明:

  • 如果题目有解,该答案即为唯一答案。
  • 输入数组均为非空数组,且长度相同。
  • 输入数组中的元素均为非负数。

示例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 可为起始索引。

示例2:

输入:
gas = [2,3,4] cost = [3,4,3] 输出: -1
解释:
你不能从 0 号或 1 号加油站出发,因为没有足够的汽油可以让你行驶到下一个加油站。
我们从 2 号加油站出发,可以获得 4 升汽油。 此时油箱有 = 0 + 4 = 4 升汽油
开往 0 号加油站,此时油箱有 4 - 3 + 2 = 3 升汽油
开往 1 号加油站,此时油箱有 3 - 3 + 3 = 3 升汽油
你无法返回 2 号加油站,因为返程需要消耗 4 升汽油,但是你的油箱只有 3 升汽油。
因此,无论怎样,你都不可能绕环路行驶一周。

解法

  1. 解法一:我们从0出发,每经过一个加油站计算剩余的油量(允许负值),那么假设有一个加油站可以从这里出发环行一圈,那么剩余的油量在这之前应该是达到了最低值,后面渐渐才“好起来了”(有起势)。
class Solution {
    public int canCompleteCircuit(int[] gas, int[] cost) {
        int minLeft = Integer.MAX_VALUE;
        int min = -1;
        int gasLeft = 0;
        for(int i = 0;i<gas.length;i++){
            gasLeft += gas[i];
            gasLeft -= cost[i];
            if(gasLeft < minLeft) {
                minLeft = gasLeft;
                min = i;
            }
        }
        int station = min+1==gas.length?0:min+1;
        return gasLeft>=0?station:-1;
    }
}
func canCompleteCircuit(gas []int, cost []int) int {
    gasLeft := 0
    min := math.MaxInt
    minIndex := 0

    for i := 0; i < len(gas); i ++ {
        gasLeft = gasLeft + gas[i] - cost[i]
        if gasLeft < min {
            min = gasLeft
            minIndex = i
        }
    }
    // 从来没有缺过油,那么从第一个开始就行
    if min >= 0 {
        return 0
    }
    // 全程下来油还是不够,则不存在这个解
    if gasLeft < 0{
        return -1
    }
    // 到哪个油站到达了历史低点,则从后面那个油站开始就一定行
    if minIndex == len(gas) -1 {
        return 0
    }
    return minIndex+1
}
  1. 解法二:一次遍历,假如有a b c d e五个加油站,如果从a出发能过b、c,但是过不了d,那么从b出发同样也不能过d;那么只需要从第一个加油站开始检查是否能作为出发点,当遇到过不了的加油站时,直接从下一个加油站开始检查。这样每个加油站最多只需遍历两次,时间复杂度仍是O(n)
func canCompleteCircuit(gas []int, cost []int) int {
    return checkIf(gas, cost, 0, 1)
}

func checkIf(gas []int, cost []int, startIndex int, round int) int {
    if round > 1 && startIndex >= 0{
        return -1
    }
    index := startIndex
    count := 0
    gasLeft := 0
    for count < len(gas){
        gasLeft = gasLeft + gas[index] - cost[index]
        index ++
        count ++
        if index == len(gas){
            index = 0
            round ++
        }
        if gasLeft < 0{
            return checkIf(gas, cost, index, round)
        }
    }
    return startIndex
}

原创文章,作者:彭晨涛,如若转载,请注明出处:https://www.codetool.top/article/leetcode134-%e5%8a%a0%e6%b2%b9%e7%ab%99/

(0)
彭晨涛彭晨涛管理者
上一篇 2020年2月15日
下一篇 2020年2月16日

相关推荐

  • leetcode403-青蛙过河

    原题 一只青蛙想要过河。 假定河流被等分为 x 个单元格,并且在每一个单元格内都有可能放有一石子(也有可能没有)。 青蛙可以跳上石头,但是不可以跳入水中。 给定石子的位置列表(用单…

    算法 2020年6月16日
    04710
  • leetcode380-常数时间插入、删除和获取随机元素

    原题 设计一个支持在平均 时间复杂度 O(1) 下,执行以下操作的数据结构。 insert(val):当元素 val 不存在时,向集合中插入该项。 remove(val):元素 v…

    2019年12月28日
    0110
  • leetcode344-反转字符串

    原题 编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组char[]的形式给出。 不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间…

    算法 2019年11月18日
    0150
  • leetcode153-寻找旋转排序数组中的最小值

    原题 假设按照升序排序的数组在预先未知的某个点上进行了旋转。 ( 例如,数组 [0,1,2,4,5,6,7] 可能变为 [4,5,6,7,0,1,2] )。 搜索一个给定的目标值,…

    算法 2020年1月3日
    080
  • leetcode704-二分查找

    原题 给定一个 n 个元素有序的(升序)整型数组 nums 和一个目标值 target ,写一个函数搜索 nums 中的 target,如果目标值存在返回下标,否则返回 -1。 示…

    算法 2019年12月29日
    080
  • leetcode260-只出现一次的数字III

    原题 给定一个整数数组 nums,其中恰好有两个元素只出现一次,其余所有元素均出现两次。 找出只出现一次的那两个元素。 示例: 输入: [1,2,1,3,2,5] 输出: [3,5…

    算法 2020年4月28日
    0150
  • leetcode142-环形链表II

    原题 给定一个链表,返回链表开始入环的第一个节点。 如果链表无环,则返回 null。 为了表示给定链表中的环,我们使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始…

    2019年12月14日
    0440
  • leetcode25-K个一组翻转链表

    原题 给你一个链表,每 k 个节点一组进行翻转,请你返回翻转后的链表。 k 是一个正整数,它的值小于或等于链表的长度。 如果节点总数不是 k 的整数倍,那么请将最后剩余的节点保持原…

    算法 2020年5月9日
    0540
  • leetcode341-扁平化嵌套列表迭代器

    原题 给定一个嵌套的整型列表。设计一个迭代器,使其能够遍历这个整型列表中的所有整数。 列表中的项或者为一个整数,或者是另一个列表。 示例1: 输入: [[1,1],2,[1,1]]…

    算法 2020年1月27日
    050
  • leetcode26-删除排序数组中的重复项

    原题 给定一个排序数组,你需要在原地删除重复出现的元素,使得每个元素只出现一次,返回移除后数组的新长度。 不要使用额外的数组空间,你必须在原地修改输入数组并在使用 O(1) 额外空…

    算法 2019年11月23日
    0210

发表回复

登录后才能评论