一.买卖股票的最佳时机
问题概述:
给定一个数组 prices ,它的第 i 个元素 prices[i] 表示一支给定股票第 i 天的价格。
你只能选择 某一天 买入这只股票,并选择在 未来的某一个不同的日子 卖出该股票。设计一个算法来计算你所能获取的最大利润。
返回你可以从这笔交易中获取的最大利润。如果你不能获取任何利润,返回 0 。

思路:
我站在每天的角度去看,总是拿历史到现在最低的价格买入,我看看今天的价格卖出,是不是比之前的历史的都要赚,如果都要赚我就卖,如果不是,我就不卖。
代码:
java
class Solution {
public int maxProfit(int[] prices) {
int minprice=Integer.MAX_VALUE;
int maxProfit=0;
for(int price:prices){
minprice=Math.min(price,minprice);
maxProfit=Math.max(maxProfit,price-minprice);
}
return maxProfit;
}
}
二.跳跃游戏
问题概述:
给你一个非负整数数组 nums ,你最初位于数组的 第一个下标 。数组中的每个元素代表你在该位置可以跳跃的最大长度。
判断你是否能够到达最后一个下标,如果可以,返回 true ;否则,返回 false 。

思路:
01234
23114
24348
第一个数最大能跳到索引2,那索引2之前的最大数为4,那他最大再变更为4,那4>=4已经含盖了最后个索引,所以为true.
01234
32104
33334
第一个数最大能跳到索引3,那索引3之前的最大数还是3,那他跳不到4.为false
那我们就从第一个数开始遍历,每次跳1格计算当前能跳到的最远位置,当遍历的数>当前最远位置,那就没法跳了,就返回false。如果当前跳到的最远位置已经大于等于最后个位置,所以已经具备了跳到最后的能力,直接返回false。
代码最后一个return true永远不会执行,但不写会编译错误。
代码:
java
class Solution {
public boolean canJump(int[] nums) {
int maxReach=0;
int n=nums.length;
for(int i=0;i<n;i++){
if(i>maxReach){
return false;
}
maxReach=Math.max(maxReach,nums[i]+i);
if(maxReach>=n-1){
return true;
}
}
return true;
}
}
三.跳跃游戏II
问题概述:
上一题问能不能跳到最后,现在问的是能跳到最后,最少只要跳几次。(题目保证一定能跳到最后)
思路:

倒着推,从左往右找第一个能跳到最后那个索引位置记作pos1,step++。再以pos1为目标位置,从左到右找第一个能跳到pos1的下一个pos2,step++,依次类推有几个pos就是最少跳跃次数。
那可能你会有疑惑,有木有可能跳不到pos1呢,其实你可以结合上题,它是从第一个数开始遍历的,能到最后,那期间的那些数都是可以跳的,那题目保证了从0能跳到最后n,那0到n这个区间一定包括pos1,那pos1左边必有能到它的数,那有人问呢,那pos2呢,这不就是n取pos1嘛,那不也就跳的到嘛。
代码:
java
class Solution {
public int jump(int[] nums) {
int pos=nums.length-1;
int step=0;
while(pos>0){
for(int i=0;i<pos;i++){
if(nums[i]+i>=pos){
pos=i;
step++;
break;
}
}
}
return step;
}
}
四.划分字母区间
问题概述:
给你一个字符串 s 。我们要把这个字符串划分为尽可能多的片段,同一字母最多出现在一个片段中。例如,字符串 "ababcc" 能够被分为 ["abab", "cc"],但类似 ["aba", "bcc"] 划分是非法的,因为b在两个字符串都出现了。

思路:
如果'a'出现的位置为2,8,10,那是不是包含'a'的片段一定得分割到10,那前面如果有'b',位置为3,11那是不是就又得切到11这个位置。
那我们就可以想到,先把每个字母最后的位置记录下来,一个一个位置切下去,看到哪才能断。

代码:
java
class Solution {
public List<Integer> partitionLabels(String s) {
List<Integer> res=new ArrayList<Integer>();
int start=0,end=0;
int[] last=new int[26];
for(int i=0;i<s.length();i++){
last[s.charAt(i)-'a']=i;
}
for(int i=0;i<s.length();i++){
end=Math.max(end,last[s.charAt(i)-'a']);
if(i==end){
res.add(end-start+1);
start=i+1;
}
}
return res;
}
}