33. 搜索旋转排序数组 - 中等
整数数组 nums 按升序排列,数组中的值 互不相同 。
在传递给函数之前,nums 在预先未知的某个下标 k(0 <= k < nums.length)上进行了 旋转,使数组变为 [nums[k], nums[k+1], …, nums[n-1], nums[0], nums[1], …, nums[k-1]](下标 从 0 开始 计数)。例如, [0,1,2,4,5,6,7] 在下标 3 处经旋转后可能变为 [4,5,6,7,0,1,2] 。
给你 旋转后 的数组 nums 和一个整数 target ,如果 nums 中存在这个目标值 target ,则返回它的下标,否则返回 -1 。
解法一:
直接遍历查找
class Solution { public int search(int[] nums, int target) { int len = nums.length; int i=0; while(i<len){ if(nums[i] == target){ return i; } i++; } return -1; } }
解法二:
二分查找,把数组分成左右两半部分,判断哪一部分是有序的,然后在有序的部分判断target是否在范围内,在 就 继续二分缩小范围,不在就到另外一半部分里查找。
class Solution { public int search(int[] nums, int target) { int right = nums.length-1; //右指针 if(right<0) return -1; if(right == 0) return nums[0]==target? 0:-1; int left = 0; //左指针 int mid = 0; //中位数下标 while(left<=right){ mid = (left+right)/2; //计算中位数 if(nums[mid] == target) return mid; if(nums[left] <= nums[mid]){ //左半边有序 if(target >= nums[left] && target < nums[mid]){ //target在左半边 right = mid - 1; }else{ left = mid + 1; } }else{ //右半边有序 if(target > nums[mid] && target <= nums[right]){//target在右半边 left = mid + 1; }else{ right = mid - 1; } } } return -1; } }
34. 在排序数组中查找元素的第一个和最后一个位置 - 中等 - 9/18
给定一个按照升序排列的整数数组 nums,和一个目标值 target。找出给定目标值在数组中的开始位置和结束位置。
如果数组中不存在目标值 target,返回 [-1, -1]。
进阶:
解析:
l 指针掌管左边蓝色区域, rr 指针掌管右边红色区域,两者互不冲突,通过不断向目标元素靠近扩大掌管区域,直到两者掌管区域接壤。
二分查找起始状态:
二分查找终止状态:
< 、≤ 、≥ 、> 目标元素对应的蓝红区域划分
总结模板:
class Solution { public int[] searchRange(int[] nums, int target) { //左边界 int leftIdx = binarySearchLeft(nums, target); //右边界 int rightIdx = binarySearchRight(nums, target); //判断是否符合 if (leftIdx <= rightIdx && rightIdx < nums.length && nums[leftIdx] == target && nums[rightIdx] == target) { return new int[]{leftIdx, rightIdx}; } return new int[]{-1, -1}; } //找左边界的二分查找 public int binarySearchLeft(int[] nums, int target) { //注意左右指针的初始值 int left = -1, right = nums.length; while (left+1 != right) { int mid = (left + right) / 2; if (nums[mid] >= target) { right = mid; } else { left = mid; } } return right; } //找右边界的二分查找 public int binarySearchRight(int[] nums, int target) { //注意左右指针的初始值 int left = -1, right = nums.length; while (left+1 != right) { int mid = (left + right) / 2; if (nums[mid] <= target) { left = mid; } else { right = mid; } } return left; } }
时间复杂度:O(log(n))。
空间复杂度:O(1)。
39. 组合总和 - 中等 - 9/22
给定一个无重复元素的正整数数组 candidates 和一个正整数 target ,找出 candidates 中所有可以使数字和为目标数 target 的唯一组合。
candidates 中的数字可以无限制重复被选取。如果至少一个所选数字数量不同,则两种组合是唯一的。
对于给定的输入,保证和为 target 的唯一组合数少于 150 个。
解析:
先向前列举所有情况,得到一个解或者走不通的时候就回溯。
class Solution { List<List<Integer>> res; public List<List<Integer>> combinationSum(int[] candidates, int target) { //存放结果 res = new ArrayList<>(); //排序 Arrays.sort(candidates); //回溯 backtrack(candidates,target,new ArrayList<>(),0); return res; } //回溯 private void backtrack(int[] candidates,int remains,List<Integer> path,int start){ //如果做减法后剩下0,则把这条路径添加进来 if(remains == 0){ res.add(new ArrayList<>(path)); return; } //遍历 for(int i=start; i<candidates.length; i++){ //如果剩余值小于0,就返回 if(remains-candidates[i] < 0) return; //剪枝 即去掉相同数的路径 如[2,2,3,7] 去掉第二个2。 if(i > 0 && candidates[i] == candidates[i-1]) continue; //添加元素到路径 path.add(candidates[i]); //回溯 backtrack(candidates, remains-candidates[i], path, i); //找到了一个解或者 remains < 0 了,将当前数字移除,然后继续尝试 path.remove(path.size()-1); } } }
42. 接雨水 - 困难 - 9/23
给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水。
方法一:双指针
class Solution { public int trap(int[] height) { int left = 0, right = height.length - 1; //定义左右指针 int ans = 0; //结果 int left_max = 0, right_max = 0; //左右当前的最大值 while(left<right){ if(height[left] < height[right]){ //如果左边高度比右边小 if (height[left] >= left_max){ //如果左边当前值 大于等于 左边最大值,则更新左边最大值 left_max = height[left]; } else { //否则 就可累加计算雨水量 ans += (left_max - height[left]); } ++left; //移动左指针 } else { //如果左边高度比右边大 if(height[right] >= right_max){ //如果右边当前值 大于等于 右边最大值,则更新右边最大值 right_max = height[right]; } else { //否则 就可累加计算雨水量 ans += (right_max - height[right]); } --right; //移动右指针 } } return ans; } }
时间复杂度:O(n)。
空间复杂度:O(1)。
方法二:动态规划
class Solution { public int trap(int[] height) { int len = height.length; //长度 if(len == 0){ return 0; } //从左往右遍历,记录左边最高值 int[] left_max = new int[len]; left_max[0] = height[0]; for(int i=1; i<len; i++){ left_max[i] = Math.max(left_max[i-1],height[i]); } //从右往左遍历,记录右边最高值 int[] right_max = new int[len]; right_max[len-1] = height[len-1]; for(int j=len-2; j>=0; j--){ right_max[j] = Math.max(right_max[j+1],height[j]); } //计算接水量 int sum = 0; for(int k=0; k<len; k++){ sum += Math.min(left_max[k],right_max[k]) - height[k]; } return sum; } }
时间复杂度:O(n),其中n是数组height的长度。计算数组leftMax和rightMax的元素值各需要遍历数组height一次,计算能接的雨水总量还需要遍历一次。
空间复杂度:O(n),其中n是数组height的长度。需要创建两个长度为n的数组leftMax 和 rightMax。
46. 全排列 - 中等
给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。
解析:回溯法
利用递归每次向 temp 里添加一个数字,数字添加够以后再回来进行回溯,再向后添加新的解。
可以理解成一层一层的添加,每一层都是一个 for 循环。
每调用一层就进入一个 for 循环,相当于列出了所有解,然后挑选了我们需要的。其实本质上就是深度优先遍历 DFS。
class Solution { public List<List<Integer>> permute(int[] nums) { List<List<Integer>> res = new ArrayList<>(); //存放结果 backtrack(nums,res,new ArrayList<>()); //回溯 return res; //返回结果 } private void backtrack(int[] nums,List<List<Integer>> res, List<Integer> tempList){ if(tempList.size() == nums.length){ //如果临时列表长度 等于 数组的长度 就添加并返回 res.add(new ArrayList<>(tempList)); return; } for(int i=0; i<nums.length; i++){ if(tempList.contains(nums[i])) continue; //如果已经存在元素,跳过 tempList.add(nums[i]); //添加元素 backtrack(nums,res,tempList); //回溯 向后继续添加 tempList.remove(tempList.size() - 1); //将 tempList 刚添加的元素,去掉,尝试新的元素 } } }
分析:
递归之前 => [1] 递归之前 => [1, 2] 递归之前 => [1, 2, 3] 递归之后 => [1, 2] 递归之后 => [1] 递归之前 => [1, 3] 递归之前 => [1, 3, 2] 递归之后 => [1, 3] 递归之后 => [1] 递归之后 => [] 递归之前 => [2] 递归之前 => [2, 1] 递归之前 => [2, 1, 3] 递归之后 => [2, 1] 递归之后 => [2] 递归之前 => [2, 3] 递归之前 => [2, 3, 1] 递归之后 => [2, 3] 递归之后 => [2] 递归之后 => [] 递归之前 => [3] 递归之前 => [3, 1] 递归之前 => [3, 1, 2] 递归之后 => [3, 1] 递归之后 => [3] 递归之前 => [3, 2] 递归之前 => [3, 2, 1] 递归之后 => [3, 2] 递归之后 => [3] 递归之后 => [] 输出的结果:[[1, 2, 3], [1, 3, 2], [2, 1, 3], [2, 3, 1], [3, 1, 2], [3, 2, 1]]