LeetCode 939.最小面积矩形

前言

最近公司比较闲,那么自己也找点事情做。这道题呢在我写这篇文章的时候谷歌、百度上都没有答案,于是乎自己就来解答一下。

题目

最小面积矩形
链接可能点不进去,所以我把完整的题目写在了下面。

描述:给定在 xy 平面上的一组点,确定由这些点组成的矩形的最小面积,其中矩形的边平行于 x 轴和 y 轴。
如果没有任何矩形,就返回0。

示例 1:
输入:[[1,1],[1,3],[3,1],[3,3],[2,2]]
输出:4
示例 2:
输入:[[1,1],[1,3],[3,1],[3,3],[4,1],[4,3]]
输出:2

提示:
1 <= points.length <= 500
0 <= points[i][0] <= 40000
0 <= points[i][1] <= 40000
所有的点都是不同的。

题意很容易理解,就是找到所有能构成矩形的4个点,然后求出它们的最小面积。示例1,最小面积为4由[[1,1],[1,3],[3,1],[3,3]]4个点构成;示例2,面积为2由[[3,1],[3,3],[4,1],[4,3]]4个点构成。

分析

看了题就知道这个题的难点在哪里:如何找到能构成矩形的4个点。于是乎我经历了下面的解题过程。

Time Limit Exceeded

首先能直接想到的就是穷举了,代码也是出奇的简单。

  int minAreaRect(vector<vector<int>>& points) {
        int minAreaValue = INT_MAX;
        //暴力穷举
        for (int i = 0; i < points.size() - 3; i ++) {
            for (int j = i + 1; j < points.size() - 2; j ++) {
                for (int k = j + 1; k < points.size() - 1; k ++) {
                    for (int l = k + 1; l < points.size(); l ++) {
                        int value = areaRect(points[i],points[j],
                                             points[k],points[l]);
                        if(value == -1) {
                            continue;
                        }
                        minAreaValue = min(minAreaValue,value);
                    }
                }
            }
        }
        return minAreaValue == INT_MAX ? 0 : minAreaValue;
    }
    //这4个点能不能构成矩形
    int areaRect(vector<int> one,vector<int> two,
                 vector<int> three,vector<int> four) {
        ///我就不贴代码了,说一下思路:
        ///1:统计4个点的x、y是否分别是两个值,并且分别都出现了2次;
        ///2:得出面积。
    }

这个题的难度是中等,所以不可能让暴力求解AC的。
暴力求解无法AC

Accept

于是乎我们转变一下思路,借助于这道题,我想到的方法是:我先固定两个存在的x1、x2,然后在两个x1、x2上找对应相等的每对y1、y2即可,那么x1、x2和每对y1、y2这4个点肯定能构成矩形。
比如:[[1,3],[3,1],[3,3],[1,3],[1,2]]。
我们发现只有两个x:1、3,然后我们在x为1和3之上去找y,发现有两个相等的y:1、3,那么答案也就是abs(x2 - x1) * abs(y2 - y1) = 4了。

        int minValue = INT_MAX;
        //x为key,相同x的y为value
        unordered_map<int, set<int>> pointMap;
        unordered_map<int, set<int>>::iterator mapIterator,tempIterator;
        for (int index = 0; index < points.size(); index ++) {
            pointMap[points[index][0]].insert(points[index][1]);
        }
        mapIterator = pointMap.begin();
        set<int>::iterator setIterator;
//        while (mapIterator != pointMap.end()) {
//            cout<<"key:"<<mapIterator->first<<" ";
//            set<int> numbers = mapIterator->second;
//            setIterator = numbers.begin();
//            while (setIterator != numbers.end()) {
//                cout<<*setIterator<<" ";
//                setIterator ++;
//            }
//            cout<<endl;
//            mapIterator ++;
//        }
        
        mapIterator = pointMap.begin();
       //双重for循环固定x1、x2
        for (; mapIterator != pointMap.end(); mapIterator ++) {
            tempIterator = mapIterator;tempIterator ++;
            for (; tempIterator != pointMap.end(); tempIterator ++) {
                int leftX = mapIterator->first;
                int rightX = tempIterator->first;
                //对两个x对应的y进行遍历,找到交集
                set<int> leftYs = mapIterator->second;
                set<int> rightYs = tempIterator->second;
                //现在我们需要从leftYs和rightYs中找到交集
                set<int> unionSet;
                setIterator = leftYs.begin();
                while (setIterator != leftYs.end()) {
                    if(rightYs.find(*setIterator) != rightYs.end()) {
                        unionSet.insert(*setIterator);
                    }
                    setIterator ++;
                }
                //如果没有两个数相等,则直接下一个循环,说明不存在y1、y2
                if(unionSet.size() < 2) {
                    continue;
                }
                //找到y1、y2的最小差值
                int minSubValue = INT_MAX;
                for (setIterator = unionSet.begin(); setIterator != unionSet.end(); setIterator ++) {
                    set<int>::iterator temp = setIterator; temp ++;
                    if(temp == unionSet.end()) { break; }
                    minSubValue = min(minSubValue,*temp - *setIterator);
                }
                minValue = min(minValue,minSubValue * abs(leftX - rightX));
            }
        }
        return minValue == INT_MAX ? 0 : minValue;

这个代码还是很通俗易懂的,当然了还有可以优化的地方,可以用位运算来做。

后记

数据结构、算法和设计模式作为软件开发的三大基础,做算法也就相当于同时练习了数据结构和算法,意义还是很大的。有些算法题看到题目后确实一点思路都没有,不过不要慌,算法也是一个不断学习的过程。时间长了,你也就会做了。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,185评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,445评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,684评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,564评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,681评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,874评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,025评论 3 408
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,761评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,217评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,545评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,694评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,351评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,988评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,778评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,007评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,427评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,580评论 2 349

推荐阅读更多精彩内容