数组


概述

数组是存放在连续内存空间上的相同类型数据的集合

可以用来解决问题的方法:

  • 双指针法
  • 二分法
  • 滑动窗口
  • 贪心算法
  • 哈希表

有序数组索引

Q:从排序数组中找到目标值并返回其索引

 输入: [1,3,5,6], 5  
 输出: 2

关键点:

  • 有序数组
  • 边界处理
int searchInsert1(int[] nums, int target) {
    int n = nums.length();
    int left = 0;
    int right = n - 1; // 定义target在左闭右闭的区间里,[left, right] 
    while (left <= right) { // 当left==right,区间[left, right]依然有效
        int middle = left + ((right - left) / 2);// 防止溢出 等同于(left + right)/2
        if (nums[middle] > target) {
            right = middle - 1; // target 在左区间,所以[left, middle - 1]
        } else if (nums[middle] < target) {
            left = middle + 1; // target 在右区间,所以[middle + 1, right]
        } else { // nums[middle] == target
            return middle;
        }
    }
    // 分别处理如下四种情况
    // 目标值在数组所有元素之前  [0, -1]
    // 目标值等于数组中某一个元素  return middle;
    // 目标值插入数组中的位置 [left, right],return  right + 1
    // 目标值在数组所有元素之后的情况 [left, right], return right + 1
    return right + 1;

int searchInsert2(int[] nums, int target) {
    int n = nums.length();
    int left = 0;
    int right = n; // 定义target在左闭右开的区间里,[left, right)  target
    while (left < right) { // 因为left == right的时候,在[left, right)是无效的空间
        int middle = left + ((right - left) >> 1);
        if (nums[middle] > target) {
             right = middle; // target 在左区间,在[left, middle)中
        } else if (nums[middle] < target) {
            left = middle + 1; // target 在右区间,在 [middle+1, right)中
        } else { // nums[middle] == target
            return middle; // 数组中找到目标值的情况,直接返回下标
        }
    }
    // 分别处理如下四种情况
    // 目标值在数组所有元素之前 [0,0)
    // 目标值等于数组中某一个元素 return middle
    // 目标值插入数组中的位置 [left, right) ,return right 即可
    // 目标值在数组所有元素之后的情况 [left, right),return right 即可
    return right;
}

原地删除元素

Q:给你一个数组 nums 和一个值 val,你需要 原地 移除所有数值等于 val 的元素,并返回移除后数组的新长度?

给定 nums = [3,2,2,3], val = 3
函数应该返回新的长度 2, 并且 nums 中的前两个元素均为 2

关键点:

  • 快慢指针
int removeElement(int[] nums, int val) {
    int slowIndex = 0; 
    for (int fastIndex = 0; fastIndex < nums.length(); fastIndex++) {  
        if (val != nums[fastIndex]) { 
            nums[slowIndex++] = nums[fastIndex]; 
        }
    }
    return slowIndex;
}

最小子串

Q:给定一个含有 n 个正整数的数组和一个正整数 s ,找出该数组中满足其和 ≥ s 的长度最小的 连续 子数组,并返回其长度。如果不存在符合条件的子数组,返回 0

输入:s = 7, nums = [2,3,1,2,4,3]
输出:2
解释:子数组 [4,3] 是该条件下的长度最小的子数组。

关键点:

  • 滑动窗口:所谓滑动窗口,「就是不断的调节子序列的起始位置和终止位置,从而得出我们要想的结果」
int minSubArrayLen(int s, vector<int>& nums) {
    int result = INT32_MAX;
    int sum = 0; // 滑动窗口数值之和
    int i = 0; // 滑动窗口起始位置
    int subLength = 0; // 滑动窗口的长度
    for (int j = 0; j < nums.size(); j++) {
        sum += nums[j];
        // 注意这里使用while,每次更新 i(起始位置),并不断比较子序列是否符合条件
        while (sum >= s) {
            subLength = (j - i + 1); // 取子序列的长度
            result = result < subLength ? result : subLength;
            sum -= nums[i++]; // 这里体现出滑动窗口的精髓之处,不断变更i(子序列的起始位置)
        }
    }
    // 如果result没有被赋值的话,就返回0,说明没有符合条件的子序列
    return result == INT32_MAX ? 0 : result;
}

俄罗斯套娃(真题)

Q:给你一个二维整数数组 envelopes ,其中 envelopes[i] = [wi, hi] ,表示第 i 个信封的宽度和高度。当另一个信封的宽度和高度都比这个信封大的时候,这个信封就可以放进另一个信封里,如同俄罗斯套娃一样。请计算 最多能有多少个 信封能组成一组“俄罗斯套娃”信封(即可以把一个信封放到另一个信封里面)

输入:envelopes = [[5,4],[6,4],[6,7],[2,3]]
输出:3
解释:最多信封的个数为 3, 组合为: [2,3] => [5,4] => [6,7]。

关键点:

  • 分治法
public int maxEnvelopes(int[][] envelopes) {
    if (envelopes.length == 0) {
        return 0;
    }    
    int n = envelopes.length;
    Arrays.sort(envelopes, new Comparator<int[]>() {
        public int compare(int[] e1, int[] e2) {
            if (e1[0] != e2[0]) {
                return e1[0] - e2[0];
            } else {
                return e2[1] - e1[1];
            }
        }
    });
    //[[2,3],[5,4],[6,7],[6,4]]
    int[] f = new int[n];
    Arrays.fill(f, 1);
    int ans = 1;
    for (int i = 1; i < n; ++i) {
        for (int j = 0; j < i; ++j) {
            if (envelopes[j][1] < envelopes[i][1]) {
                f[i] = Math.max(f[i], f[j] + 1);
            }
        }
        ans = Math.max(ans, f[i]);
    }
    return ans;
}

最长连续递增序列

Q:给定一个未经排序的整数数组,找到最长且 连续递增的子序列,并返回该序列的长度。

输入:nums = [1,3,5,4,7]
输出:3
解释:最长连续递增序列是 [1,3,5], 长度为3。

关键点:

  • 贪心算法,找到最长的适合条件的组合
public int findLengthOfLCIS(int[] nums) {
    int ans = 0;
    int n = nums.length;
    int start = 0;
    for (int i = 0; i < n; i++) {
        if (i > 0 && nums[i] <= nums[i - 1]) {
            start = i;
        }
        ans = Math.max(ans, i - start + 1);
    }
    return ans;
}

盛最多水的容器(真题)

Q:给你 n 个非负整数 a1,a2,…,an,每个数代表坐标中的一个点 (i, ai) 。在坐标内画 n 条垂直线,垂直线 i 的两个端点分别为 (i, ai) 和 (i, 0) 。找出其中的两条线,使得它们与 x 轴共同构成的容器可以容纳最多的水。

输入:[1,8,6,2,5,4,8,3,7]
输出:49 

关键点:

  • 双指针,分别指向头尾两端,不断替换值小的数
public int maxArea(int[] height) {
    int l = 0, r = height.length - 1;
    int ans = 0;
    while (l < r) {
        int area = Math.min(height[l], height[r]) * (r - l);
        ans = Math.max(ans, area);
        if (height[l] <= height[r]) {
            ++l;
        }
        else {
            --r;
        }
    }
    return ans;
}

接雨水问题(真题)

Q:给定 n 个非负整数表示每个宽度为 1 的柱子的高度图,计算按此排列的柱子,下雨之后能接多少雨水

输入:height = [0,1,0,2,1,0,1,3,2,1,2,1]
输出:6 

关键点:

  • 双指针,分别指向头尾两端,不断替换值小的数
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;
}

和为K的子数组

Q:给定一个整数数组和一个整数 k,你需要找到该数组中和为 k 的连续的子数组的个数

输入:nums = [1,1,1], k = 2
输出: 2 , [1,1] 与 [1,1] 为两种不同的情况。

关键点:

  • 哈希表
public int subarraySum(int[] nums, int k) {
    int count = 0, pre = 0;
    HashMap < Integer, Integer > mp = new HashMap < > ();
    mp.put(0, 1);
    for (int i = 0; i < nums.length; i++) {
        pre += nums[i];
        if (mp.containsKey(pre - k)) {
            count += mp.get(pre - k);
        }
        mp.put(pre, mp.getOrDefault(pre, 0) + 1);
    }
    return count;
}