zoukankan      html  css  js  c++  java
  • 凸包问题 Graham Scan

    2020-01-09 15:14:21

    凸包问题是计算几何的核心问题,并且凸包问题的研究已经持续了好多年,这中间涌现出了一大批优秀的算法。

    凸包问题的最优解法是Graham Scan算法,该算法可以保证在最差情况下也能在O(nlogn)的时间复杂度求出结果。

    Graham Scan算法的核心思路有两个步骤:

    1. 预处理:将所有的点按照某个给定的点根据夹角进行排序;

    2. 在排序好的结果上,按照顺序逆时针依次进行判断(to left test),将符合条件的节点加入栈中;

    在具体实现的时候有一个非常好用的trick,就是不直接对夹角进行排序,而是假定在 +inf 和 -inf 有两个点,那么我们只需要根据x轴进行排序,就可以得到它们相对于两个无穷远点的夹角排序。

    这样我们就可以直接使用第二步的递推迭代得到upper hull和lower hull,最后将这两个结果拼接起来就是最终的结果。

    下面根据一条leetcode的题目来具体看下代码实现。

    问题描述

     

    问题求解

        public int[][] outerTrees(int[][] points) {
            if (points.length <= 3) return points;
            int n = points.length;
            Arrays.sort(points, new Comparator<int[]>(){
                public int compare(int[] o1, int[] o2) {
                    return o1[0] == o2[0] ? o1[1] - o2[1] : o1[0] - o2[0];
                }
            });
            Stack<int[]> upper = new Stack<>();
            Stack<int[]> lower = new Stack<>();
            Stack<int[]> left = new Stack<>();
            
            upper.add(points[n - 1]);
            upper.add(points[n - 2]);
            lower.add(points[0]);
            lower.add(points[1]);
            
            for (int i = 0; i < n - 2; i++) left.add(points[i]);
            while (!left.isEmpty()) {
                int[] curr = left.pop();
                if (upper.size() == 1) {
                    upper.add(curr);
                    continue;
                }
                int[] p2 = upper.pop();
                int[] p1 = upper.pop();
                if (to_left_test(p1, p2, curr)) {
                    upper.add(p1);
                    upper.add(p2);
                    upper.add(curr);
                }
                else {
                    upper.add(p1);
                    left.add(curr);
                }
            }
            
            for (int i = n - 1; i > 1; i--) left.add(points[i]);
            while (!left.isEmpty()) {
                int[] curr = left.pop();
                if (lower.size() == 1) {
                    lower.add(curr);
                    continue;
                }
                int[] p2 = lower.pop();
                int[] p1 = lower.pop();
                if (to_left_test(p1, p2, curr)) {
                    lower.add(p1);
                    lower.add(p2);
                    lower.add(curr);
                }
                else {
                    lower.add(p1);
                    left.add(curr);
                }
            }
            
            Set<int[]> set = new HashSet<>();
            set.addAll(upper);
            set.addAll(lower);
            int[][] res = new int[set.size()][2];
            int idx = 0;
            for (int[] p : set) {
                res[idx][0] = p[0];
                res[idx][1] = p[1];
                idx += 1;
            }
            return res;
        }
        
        
        private boolean to_left_test(int[] p1, int[] p2, int[] p3) {
            return p1[0] * p2[1] + p1[1] * p3[0] + p2[0] * p3[1] -
            p2[1] * p3[0] - p1[1] * p2[0] - p1[0] * p3[1] >= 0;
        }
    

      

  • 相关阅读:
    javascript 读取内联之外的样式(style、currentStyle、getComputedStyle区别介绍) (转载)
    JS笔记2 --定义对象
    JS笔记1
    Chrome 中的 JavaScript 断点设置和调试技巧 (转载)
    屏蔽移动端浏览器的长按事件
    移除IOS下按钮的原生样式
    HTML5中的Range对象的研究(转载)
    js中的 window.location、document.location、document.URL 对像的区别(转载)
    html中插入flash代码详解(转载)
    关于获取各种浏览器可见窗口大小(转载)
  • 原文地址:https://www.cnblogs.com/hyserendipity/p/12171662.html
Copyright © 2011-2022 走看看