Java实现 LeetCode 699 掉落的方块(线段树?)

谁借莪1个温暖的怀抱¢ 2023-07-23 05:57 42阅读 0赞

699. 掉落的方块

在无限长的数轴(即 x 轴)上,我们根据给定的顺序放置对应的正方形方块。

第 i 个掉落的方块(positions[i] = (left, side_length))是正方形,其中 left 表示该方块最左边的点位置(positions[i][0]),side_length 表示该方块的边长(positions[i][1])。

每个方块的底部边缘平行于数轴(即 x 轴),并且从一个比目前所有的落地方块更高的高度掉落而下。在上一个方块结束掉落,并保持静止后,才开始掉落新方块。

方块的底边具有非常大的粘性,并将保持固定在它们所接触的任何长度表面上(无论是数轴还是其他方块)。邻接掉落的边不会过早地粘合在一起,因为只有底边才具有粘性。

返回一个堆叠高度列表 ans 。每一个堆叠高度 ans[i] 表示在通过 positions[0], positions[1], …, positions[i] 表示的方块掉落结束后,目前所有已经落稳的方块堆叠的最高高度。

示例 1:

  1. 输入: [[1, 2], [2, 3], [6, 1]]
  2. 输出: [2, 5, 5]
  3. 解释:
  4. 第一个方块 positions[0] = [1, 2] 掉落:
  5. _aa
  6. _aa
  7. -------
  8. 方块最大高度为 2
  9. 第二个方块 positions[1] = [2, 3] 掉落:
  10. __aaa
  11. __aaa
  12. __aaa
  13. _aa__
  14. _aa__
  15. --------------
  16. 方块最大高度为5
  17. 大的方块保持在较小的方块的顶部,不论它的重心在哪里,因为方块的底部边缘有非常大的粘性。
  18. 第三个方块 positions[1] = [6, 1] 掉落:
  19. __aaa
  20. __aaa
  21. __aaa
  22. _aa
  23. _aa___a
  24. --------------
  25. 方块最大高度为5
  26. 因此,我们返回结果[2, 5, 5]。

示例 2:

  1. 输入: [[100, 100], [200, 100]]
  2. 输出: [100, 100]
  3. 解释: 相邻的方块不会过早地卡住,只有它们的底部边缘才能粘在表面上。

注意:

1 <= positions.length <= 1000.
1 <= positions[i][0] <= 10^8.
1 <= positions[i][1] <= 10^6.

PS:
按照左端点放进树

  1. import java.util.*;
  2. class Solution {
  3. // 描述方块以及高度
  4. private class Node {
  5. int l, r, h, maxR;
  6. Node left, right;
  7. public Node(int l, int r, int h, int maxR) {
  8. this.l = l;
  9. this.r = r;
  10. this.h = h;
  11. this.maxR = maxR;
  12. this.left = null;
  13. this.right = null;
  14. }
  15. }
  16. //
  17. public List<Integer> fallingSquares(int[][] positions) {
  18. // 创建返回值
  19. List<Integer> res = new ArrayList<>();
  20. // 根节点,默认为零
  21. Node root = null;
  22. // 目前最高的高度
  23. int maxH = 0;
  24. for (int[] position : positions) {
  25. int l = position[0]; // 左横坐标
  26. int r = position[0] + position[1]; // 右横坐标
  27. int e = position[1]; // 边长
  28. int curH = query(root, l, r); // 目前区间的最高的高度
  29. root = insert(root, l, r, curH + e);
  30. maxH = Math.max(maxH, curH + e);
  31. res.add(maxH);
  32. }
  33. return res;
  34. }
  35. private Node insert(Node root, int l, int r, int h) {
  36. if (root == null) return new Node(l, r, h, r);
  37. if (l <= root.l)
  38. root.left = insert(root.left, l, r, h);
  39. else
  40. root.right = insert(root.right, l, r, h);
  41. // 最终目标是仅仅需要根节点更新 maxR
  42. root.maxR = Math.max(r, root.maxR);
  43. return root; // 返回根节点
  44. }
  45. private int query(Node root, int l, int r) {
  46. // 新节点的左边界大于等于目前的maxR的话,直接得到0,不需要遍历了
  47. if (root == null || l >= root.maxR) return 0;
  48. // 高度
  49. int curH = 0;
  50. if (!(r <= root.l || root.r <= l)) // 是否跟这个节点相交
  51. curH = root.h;
  52. // 剪枝
  53. curH = Math.max(curH, query(root.left, l, r));
  54. if (r > root.l)
  55. curH = Math.max(curH, query(root.right, l, r));
  56. return curH;
  57. }
  58. }

发表评论

表情:
评论列表 (有 0 条评论,42人围观)

还没有评论,来说两句吧...

相关阅读