134. Gas Station

本文解析了一道经典的加油站问题,探讨如何确定能围绕环形路线完成旅行的起点。文章提供了多种算法思路,包括通过求最大连续子序列和来寻找合适的起点,并给出了具体的实现代码。

欢迎fork and star:Nowcoder-Repository-github

134. Gas Station

题目

 There are N gas stations along a circular route, where the amount of gas at station i is gas[i].

You have a car with an unlimited gas tank and it costs cost[i] of gas to travel from station i to its next station (i+1). You begin the journey with an empty tank at one of the gas stations.

Return the starting gas station's index if you can travel around the circuit once, otherwise return -1.

Note:
The solution is guaranteed to be unique. 

解析

  • 思路1:可以穷举每个位置,求和是否大于0;
  • 非常经典的一道题。可以转换成求最大连续和做,但是有更简单的方法。
  • 基于一个数学定理:如果一个数组的总和非负,那么一定可以找到一个起始位置,从他开始绕数组一圈,累加和一直都是非负的
  • 继续对问题进行抽象,我们肯定希望一开始的路程都是加油>消耗的,这样就可以累积油量应付后面的消耗量大的,根据这个思路,就转变成求数组的最大连续子序列的和(并且记录起始下标),这是我们前面接触过的题目。
  • 因为是循环数组,我们还得考虑一种情况:就是首尾都是正数,怎么办呢?
  • 正因为是循环数组,还有一种类似于快排前后指针的操作。
class Solution {
public:
    int canCompleteCircuit(vector<int> &gas, vector<int> &cost) {

        if (gas.size()==0||cost.size()==0||gas.size()!=cost.size())
        {
            return -1;
        }
        int index = 0;
        int sum_resdual = 0;

        for (int j = 0; j < gas.size();j++)
        {
            sum_resdual += (gas[j] - cost[j]);
        }
        if (sum_resdual<0)
        {
            return -1;
        }

        sum_resdual = 0;
        for (int i = 0; i < gas.size();i++)
        {
            //index = 0;  //bug
            sum_resdual += (gas[i] - cost[i]);
            if (sum_resdual<0)
            {
                index = i + 1;
                sum_resdual = 0;
            }
        }

        return index;
    }
};
  • 换个写法
int canCompleteCircuit(vector<int> &gas, vector<int> &cost) {
  int start = 0; // 起始位置
  int remain = 0; // 当前剩余燃料
  int debt = 0; // 前面没能走完的路上欠的债

  for (int i = 0; i < gas.size(); i++) {
    remain += gas[i] - cost[i];
    if (remain < 0) {
      debt += remain;
      start = i + 1;
      remain = 0;
    }
  }

  return remain + debt >= 0 ? start : -1;  //这句话保证了求和sum<0,返回-1
}
  • 从start出发, 如果油量足够, 可以一直向后走 end++; 油量不够的时候,start向后退 最终 start == end的时候,如果有解一定是当前 start所在位置

class Solution {
public:  int canCompleteCircuit(vector<int> &gas, vector<int> &cost) {      
        int start = gas.size() - 1;
        int end = 0;
        int sum = gas[start] - cost[start];
        while(start > end){
            if(sum >= 0){
                sum += gas[end] - cost[end];
                ++end;
            }else{
                --start;
                sum += gas[start] - cost[start];
            }
        }
        return sum >=0 ? start : -1;
         
         
    }
};

题目来源

### 高德地图日志输出的意义 在 Android 开发过程中,当集成高德地图功能时,可能会遇到特定的日志输出。以下是针对 `GasStation` 和 `高德地图客户端已经安装` 这两类日志的具体解释。 #### 关于 'GasStation' 如果在调试或运行应用期间发现日志中有 `'GasStation'` 输出,则可能表示当前的地图操作涉及到了加油站的相关数据点。这通常发生在以下场景中: - 地图上加载了 POI(兴趣点)数据,而这些数据包含了加油站的信息。 - 应用正在查询与加油站点有关的服务或者路径规划[^1]。 这种日志可能是由高德地图 SDK 自动记录的行为,用于指示某些特定类型的地理特征被处理或展示给用户。它并不一定代表错误或警告;相反,它是正常工作流程的一部分,表明 SDK 正确解析并呈现了相关位置信息。 #### 关于 '高德地图客户端已经安装' 此条目明确指出设备上存在官方发布的高德地图应用程序实例。这一确认过程通常是通过检查目标软件包名来完成的——对于高德而言,默认情况下其主包名称为 `com.autonavi.minimap` 或其他变体形式[^2]。一旦检测到对应的应用存在于用户的移动终端里,便会触发上述消息打印至控制台作为反馈机制之一: ```java public static boolean isGaoDeMapInstalled(Context context){ try { ApplicationInfo info = context.getPackageManager() .getApplicationInfo("com.autonavi.minimap", PackageManager.GET_META_DATA); return true; } catch (PackageManager.NameNotFoundException e) { return false; } } ``` 这段伪代码展示了如何利用 Android 提供的标准 API 来验证指定 APP 是否可用,并据此调整后续逻辑走向比如决定采用嵌入式视图还是跳转外部链接等形式提供服务[^3]。 综上所述,“高德地图客户端已经安装”的提示有助于开发者了解环境状态从而做出更优决策同时也让用户获得更好的体验效果。 ```java if(isGaoDeMapInstalled(context)){ Log.d("MAP_STATUS","高德地图客户端已经安装"); } else{ Toast.makeText(context,"请先下载安装高德地图!",Toast.LENGTH_SHORT).show(); } ``` 以上示例简单演示了基于前面提到方法的结果来进行基本交互设计的一种方式。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值