首页 > 其他分享 >Maximum Area Rectangle With Point Constraints II

Maximum Area Rectangle With Point Constraints II

时间:2024-12-09 23:11:00浏览次数:4  
标签:yCoord Area int 纵坐标 Point Maximum 横坐标 xs 矩形

Maximum Area Rectangle With Point Constraints II

There are n points on an infinite plane. You are given two integer arrays xCoord and yCoord where (xCoord[i], yCoord[i]) represents the coordinates of the ith point.

Your task is to find the maximum area of a rectangle that:

  • Can be formed using four of these points as its corners.
  • Does not contain any other point inside or on its border.
  • Has its edges parallel to the axes.

Return the maximum area that you can obtain or -1 if no such rectangle is possible.

 

Example 1:

Input: xCoord = [1,1,3,3], yCoord = [1,3,1,3]

Output: 4

Explanation:

We can make a rectangle with these 4 points as corners and there is no other point that lies inside or on the border. Hence, the maximum possible area would be 4.

Example 2:

Input: xCoord = [1,1,3,3,2], yCoord = [1,3,1,3,2]

Output: -1

Explanation:

There is only one rectangle possible is with points [1,1], [1,3], [3,1] and [3,3] but [2,2] will always lie inside it. Hence, returning -1.

Example 3:

Input: xCoord = [1,1,3,3,1,3], yCoord = [1,3,1,3,2,2]

Output: 2

Explanation:

The maximum area rectangle is formed by the points [1,3], [1,2], [3,2], [3,3], which has an area of 2. Additionally, the points [1,1], [1,2], [3,1], [3,2] also form a valid rectangle with the same area.

 

Constraints:

1 <= xCoord.length == yCoord.length <= 2 * 105

0 <= xCoord[i], yCoord[i] <= 8 * 107

All the given points are unique.

 

解题思路

  赛时想复杂了,罚坐了一个多小时都没做出来。

  考虑枚举每一个点作为矩阵的右上角(对应矩阵坐标系的右下角)。假设当前枚举到的右上角的点是 $(x_2, y_2)$,由于矩形的边要与坐标轴平行,因此矩形左上角的点的纵坐标应该是 $y_2$,那么我们从所有纵坐标是 $y_2$ 的点中,找到小于 $x_2$ 的最大横坐标(否则矩形的边上会包含其他点),记为 $x_1$。同理矩形右下角的点的横坐标应该是 $x_2$,从所有横坐标是 $x_2$ 的点中,找到小于 $y_2$ 的最大纵坐标,记为 $y_1$。这样我们也确定了矩形左下角的点 $(x_1, y_1)$。当然上述矩形存在的条件是点 $(x_2, y_2)$,$(x_1, y_2)$,$(x_2, y_1)$,$(x_1, y_1)$ 要同时存在。

  那么我们如何快速找到纵坐标是 $y_2$ 的所有点中,横坐标小于 $x_2$ 的最大值?我们可以给所有纵坐标值开一个 std::set 用来存储纵坐标相同的点的横坐标,每次查询的时候二分即可。同理给所有横坐标值开一个哈希表存储横坐标相同的点的纵坐标。另外还可以以横坐标为第一关键字,纵坐标为第二关键字来从小到大枚举每个点,并维护每个横坐标最大的纵坐标,每个纵坐标最大的横坐标。

  最后的问题是确定了矩形的左下角 $(x_1, y_1)$,右上角 $(x_2, y_2)$ 后,然后快速判定矩形内是否含有其他的点?这个可以用二维数点的做法实现。参考二维前缀和,把询问的矩形 $(x_1, y_1, x_2, y_2)$ 拆成 $4$ 个部分:$R_1: (1, 1, x_2, y_2)$,$R_2: (1, 1, x_1 - 1, y_2)$,$R_3: (1, 1, x_2, y_1 - 1)$,$R_4: (1, 1, x_1 - 1, y_1 - 1)$。用 $C(R)$ 表示矩形 $R$ 内点的数量,那么矩形 $(x_1, y_1, x_2, y_2)$ 内点的数量等价于 $C(R_1) - C(R_2) - C(R_3) + C(R_4)$。

  我们把所有询问的矩形都拆成这 $4$ 个部分,并以四元组 $(x,y,k,i)$ 的形式表示,其中 $x,y$ 表示拆分后的矩形的右上角的坐标(因为所有拆分矩形的左下角都是 $(1,1)$ 因此忽略),$k$ 是系数 $-1$ 或 $1$,对应要减去还是加上这个矩形内点的数量,$i$ 是这个拆分出来的矩形对应原本第 $i$ 个询问的矩形。然后对四元组从小到大排序,在枚举的过程中用树状数组来维护所有横坐标小于等于当前四元组对应的 $x$ 的点的纵坐标,然后在树状数组中查询所有纵坐标不超过 $y$ 的点的数量,就是这个矩形 $(1,1,x,y)$ 内点的数量。

  最后如果矩形内点的数量恰好等于 $4$(因为有 $4$ 个端点),说明这是一个合法矩形。

  AC 代码如下,时间复杂度为 $O(nm \log{nm})$:

class Solution {
public:
    long long maxRectangleArea(vector<int>& xCoord, vector<int>& yCoord) {
        int n = xCoord.size();
        set<array<int, 2>> st;
        vector<int> xs;
        for (int i = 0; i < n; i++) {
            st.insert({xCoord[i], yCoord[i]});
            xs.push_back(yCoord[i]);
        }
        sort(xs.begin(), xs.end());
        xs.erase(unique(xs.begin(), xs.end()));
        map<int, int> r, c;
        vector<array<int, 4>> p, q;
        auto find = [&](int x) {
            return lower_bound(xs.begin(), xs.end(), x) - xs.begin() + 1;
        };
        for (auto &[x, y] : st) {
            if (r.count(x) && c.count(y) && st.count({c[y], r[x]})) {
                int x1 = c[y], y1 = find(r[x]), x2 = x, y2 = find(y);
                p.push_back({x1, y1, x2, y2});
                q.push_back({x2, y2, 1, int(p.size()) - 1});
                q.push_back({x1 - 1, y2, -1, int(p.size()) - 1});
                q.push_back({x2, y1 - 1, -1, int(p.size()) - 1});
                q.push_back({x1 - 1, y1 - 1, 1, int(p.size()) - 1});
            }
            r[x] = y, c[y] = x;
        }
        sort(q.begin(), q.end());
        vector<int> s(p.size()), tr(xs.size() + 1);
        auto lowbit = [&](int x) {
            return x & -x;
        };
        auto add = [&](int x) {
            for (int i = x; i <= xs.size(); i += lowbit(i)) {
                tr[i]++;
            }
        };
        auto query = [&](int x) {
            int ret = 0;
            for (int i = x; i; i -= lowbit(i)) {
                ret += tr[i];
            }
            return ret;
        };
        for (int i = 0; i < q.size(); i++) {
            while (!st.empty() && st.begin()->at(0) <= q[i][0]) {
                add(find(st.begin()->at(1)));
                st.erase(st.begin());
            }
            s[q[i][3]] += query(q[i][1]) * q[i][2];
        }
        long long ret = -1;
        for (int i = 0; i < p.size(); i++) {
            if (s[i] == 4) ret = max(ret, 1ll * (p[i][2] - p[i][0]) * (xs[p[i][3] - 1] - xs[p[i][1] - 1]));
        }
        return ret;
    }
};

 

参考资料

  静态二维数点问题【力扣周赛 427】:https://www.bilibili.com/video/BV1YeqHYSEhK/

标签:yCoord,Area,int,纵坐标,Point,Maximum,横坐标,xs,矩形
From: https://www.cnblogs.com/onlyblues/p/18596216

相关文章

  • 举例说明pointer-events有什么实际用途?
    pointer-events在前端开发中非常实用,它控制着元素如何响应指针事件,例如鼠标点击、触摸、悬停等。以下是一些实际应用场景的例子:1.创建不可点击的区域/元素:禁用链接:假设你有一个链接,但在某些情况下你想暂时禁用它,可以使用pointer-events:none;。这将阻止用户点击链接,同......
  • [LeetCode] 2684. Maximum Number of Moves in a Grid
    Youaregivena0-indexedmxnmatrixgridconsistingofpositiveintegers.Youcanstartatanycellinthefirstcolumnofthematrix,andtraversethegridinthefollowingway:Fromacell(row,col),youcanmovetoanyofthecells:(row-1,col+......
  • CF2050F Maximum modulo equality 题解
    【题意简述】你有一个长度为\(n\)的数组\(a\)。每一次询问给定\(l,r\),寻找最大的\(m\)使得\(a_l\)到\(a_r\)的所有数对\(m\)同余,【前置数学芝士】首先是一个非常Naive的结论,请自行感性证明:设\(a>b\),\(a\)和\(b\)对\(a-b\)同余。理性证明:设\(p=a-b\),\(......
  • ORB-SLAM2源码学习:MapPoint.cc:MapPoint::UpdateNormalAndDepth()计算平均观测方向以
    前言这个函数是属于地图点属性的一部分。1.函数声明voidMapPoint::UpdateNormalAndDepth(){....}2.函数定义1.获取观测到该地图点的所有关键帧的信息 map<KeyFrame*,size_t>observations;KeyFrame*pRefKF;cv::MatPos;{unique_lock<m......
  • 随机链表的复制(java),注意NullPointerException
    题目描述:给你一个长度为 n 的链表,每个节点包含一个额外增加的随机指针 random ,该指针可以指向链表中的任何节点或空节点。构造这个链表的 深拷贝。 深拷贝应该正好由 n 个 全新 节点组成,其中每个新节点的值都设为其对应的原节点的值。新节点的 next 指针和 rand......
  • Gradient checkpointing 核心流程详细讲解
    文章目录0.概述1.简单反向传播1.1整体流程1.2详细说明1.3总结2.初步优化版本2.1整体流程2.2详细说明2.3总结3.Checkpointed反向传播3.1整体流程3.2详细说明3.3总结4.补充:内存分配算法参考0.概述Gradientcheckpointing的核心思想是不保存所有层......
  • 如何使用普通元素拥有像textarea元素一样缩放?
    要让普通元素像textarea一样可以缩放,你需要使用一些CSS技巧和JavaScript,因为HTML中没有直接的属性可以赋予普通元素textarea的缩放行为。以下是一种实现方法:1.CSS样式:.resizable-div{overflow:auto;/*允许内容溢出并显示滚动条*/resize:both;/*允许用......
  • JavaSwing JTextArea
    try{BeautyEyeLNFHelper.frameBorderStyle=BeautyEyeLNFHelper.FrameBorderStyle.osLookAndFeelDecorated;//UIManager.put("RootPane.setupButtonVisible",false);org.jb2011.lnf.beautyeye.BeautyEyeLNFHelpe......
  • 3D点云-Pointnet++模型解读(附源码+论文)
    3D点云-Pointnet++模型代码链接:pointnet2-pytorch-study(关键部分代码注释详细,参考Pointnet_Pointnet2_pytorch)论文链接:PointNet++:DeepHierarchicalFeatureLearningonPointSetsinaMetricSpace官方链接:pointnet2(源码基于TensorFlow)公开3D点云数据集:modelnet4......
  • E86 换根DP CF1324F Maximum White Subtree
    视频链接:E86换根DPCF1324FMaximumWhiteSubtree_哔哩哔哩_bilibili  MaximumWhiteSubtree-洛谷|计算机科学教育新生态//换根DPO(n)#include<bits/stdc++.h>usingnamespacestd;constintN=200005;vector<int>e[N];intn,a[N],f[N];voiddfs(int......