题目

给你一个下标从 0 开始的二维整数数组 flowers ,其中 flowers[i] = [starti, endi] 表示第 i 朵花的 花期 从 starti 到 endi (都 包含)。同时给你一个下标从 0 开始大小为 n 的整数数组 persons ,persons[i] 是第 i 个人来看花的时间。

请你返回一个大小为 n 的整数数组 answer ,其中 answer[i]是第 i 个人到达时在花期内花的 数目 。

示例 1: image.png

输入:flowers = [[1,6],[3,7],[9,12],[4,13]], persons = [2,3,7,11]
输出:[1,2,2,2]
解释:上图展示了每朵花的花期时间,和每个人的到达时间。
对每个人,我们返回他们到达时在花期内花的数目。

示例 2: image.png

输入:flowers = [[1,10],[3,3]], persons = [3,3,2]
输出:[2,2,1]
解释:上图展示了每朵花的花期时间,和每个人的到达时间。
对每个人,我们返回他们到达时在花期内花的数目。

提示:

1 <= flowers.length <= 5 10^4
flowers[i].length == 2
1 <= starti <= endi <= 10^9
1 <= persons.length <= 5
10^4
1 <= persons[i] <= 10^9

来源:力扣(LeetCode)
链接:https://leetcode-cn.com/problems/number-of-flowers-in-full-bloom
著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

思路

比较常规的思路是:正在开的花的个数 为所有在此时间节点及之前开放的花的个数减去在时间节点之前凋谢的花的个数,来自「这里」。见代码一。

也可以使用差分,见代码二。

代码

代码一 常规思路

  1. class Solution {
  2. public int[] fullBloomFlowers(int[][] flowers, int[] persons) {
  3. int n = persons.length;
  4. int m = flowers.length;
  5. int[] ans = new int[n];
  6. int[] start = new int[m];
  7. int[] end = new int[m];
  8. for (int i = 0; i < m; i++) {
  9. start[i] = flowers[i][0];
  10. end[i] = flowers[i][1];
  11. }
  12. Arrays.sort(start);
  13. Arrays.sort(end);
  14. for (int i = 0; i < n; i++) {
  15. int s = bsearch(start, persons[i] + 1);
  16. int e = bsearch(end, persons[i]);
  17. ans[i] = s - e;
  18. }
  19. return ans;
  20. }
  21. public int bsearch(int[] nums, int target) {
  22. int n = nums.length;
  23. int low = 0;
  24. int high = n;
  25. while (low < high) {
  26. int mid = low + (high - low) / 2;
  27. if (nums[mid] < target) {
  28. low = mid + 1;
  29. } else {
  30. high = mid;
  31. }
  32. }
  33. return low;
  34. }
  35. }

代码二 差分

参考「这里」。通过本题学习了差分这个知识点。差分可以看做是前缀和的逆策略,当需要将数组某一区间内的所有值都加上一个值的时候,如果一个个加会比较费时,引入差分数组后只需要更新两个值。这是差分的设计初衷。

  1. class Solution {
  2. public int[] fullBloomFlowers(int[][] flowers, int[] persons) {
  3. TreeMap<Integer, Integer> map = new TreeMap<>();
  4. for (int[] f : flowers) {
  5. map.put(f[0], map.getOrDefault(f[0], 0) + 1);
  6. map.put(f[1] + 1, map.getOrDefault(f[1] + 1, 0) - 1);
  7. }
  8. // persons数组中的值在map中不一定存在,后面会还原数组,因此这里可以将不存在的键值添加上,最后直接获取答案就好了
  9. for (int p : persons) {
  10. if (!map.containsKey(p)) {
  11. map.put(p, 0);
  12. }
  13. }
  14. // 还原前缀和数组
  15. int sum = 0;
  16. for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
  17. sum += entry.getValue();
  18. int k = entry.getKey();
  19. map.put(k, sum);
  20. }
  21. int[] ans = new int[persons.length];
  22. for (int i = 0; i < persons.length; i++) {
  23. ans[i] = map.getOrDefault(persons[i], 0);
  24. }
  25. return ans;
  26. }
  27. }

还可以这么写,个人觉得这么写更好。

  1. class Solution {
  2. public int[] fullBloomFlowers(int[][] flowers, int[] persons) {
  3. TreeMap<Integer, Integer> map = new TreeMap<>();
  4. for (int[] f : flowers) {
  5. map.put(f[0], map.getOrDefault(f[0], 0) + 1);
  6. map.put(f[1] + 1, map.getOrDefault(f[1] + 1, 0) - 1);
  7. }
  8. int sum = 0;
  9. for (Map.Entry<Integer, Integer> entry : map.entrySet()) {
  10. sum += entry.getValue();
  11. int k = entry.getKey();
  12. map.put(k, sum);
  13. }
  14. int[] ans = new int[persons.length];
  15. // 由于没有向上面那样往map添加不存在的键值,因此需要查找不大于persons[i]的键值,缺失的键值的值一定和这样查找到的值一样
  16. for (int i = 0; i < persons.length; i++) {
  17. // 注意可能返回null, 对应来的时刻一朵花没开
  18. Integer k = map.floorKey(persons[i]);
  19. if (k != null) {
  20. ans[i] = map.get(k);
  21. }
  22. }
  23. return ans;
  24. }
  25. }