凸包问题 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;
    }

  

posted @ 2020-01-09 15:14  hyserendipity  阅读(382)  评论(0编辑  收藏  举报