In-Place Planar Convex Hull Algorithms
In-Place Planar Convex Hull Algorithms
In-Place Planar Convex Hull Algorithms
Herv Brnnimann1 , John Iacono1 , Jyrki Katajainen2 , Pat Morin3 , Jason e o Morrison4 , and Godfried Toussaint3
CIS, Polytechnic University, Six Metrotech, Brooklyn, New York, 11201. {hbr,jiacono}@poly.edu 2 Department of Computing, University of Copenhagen, Universitetsparken 1, DK-2100 Copenhagen East, Denmark. jyrki@diku.dk 3 SOCS, McGill University, 3480 University St., Suite 318, Montral, Qubec, e e CANADA, H3A 2A7. {morin,godfried}@cgm.cs.mcgill.ca School of Computer Science, Carleton University, 1125 Colonel By Dr., Ottawa, Ontario, CANADA, K1S 5B6. morrison@cs.carleton.ca
1
Abstract. An in-place algorithm is one in which the output is given in the same location as the input and only a small amount of additional memory is used by the algorithm. In this paper we describe three in-place algorithms for computing the convex hull of a planar point set. All three algorithms are optimal, some more so than others. . .
Introduction
Let S = {S[0], . . . , S[n 1]} be a set of n distinct points in the Euclidean plane. The convex hull of S is the minimal convex region that contains every point of S. From this denition, it follows that the convex hull of S is a convex polygon whose vertices are points of S. For convenience, we say that a point p is on the convex hull of S if p is a vertex of the convex hull of S. As early as 1973, Graham [13] gave a convex hull algorithm with O(n log n) worst-case running time. Shamos [32] later showed that, in any model of computation where sorting has an (n log n) lower bound, every convex hull algorithm must require (n log n) time for some inputs. Despite these matching upper and lower bounds, and probably because of the many applications of convex hulls, a number of other planar convex hull algorithms have been published since Grahams algorithm [1,2,4,6,11,17,27,28,21,35]. Of particular note is the Ultimate(?) algorithm of Kirkpatrick and Seidel [21] that computes the convex hull of a set of n points in the plane in O(n log h) time, where h is the number of vertices of the convex hull. The same authors show that, on algebraic decision trees of any xed order, (n log h) is a lower bound for computing convex hulls of sets of n points having convex hulls with h vertices.
This research was partly funded by the National Science Foundation, the Natural Sciences and Engineering Research Council of Canada and the Danish Natural Science Research Council under contract 9801749 (project Performance Engineering).
Because of the importance of planar convex hulls, it is natural to try and improve the running time and storage requirements of planar convex hull algorithms. In this paper, we focus on reducing the intermediate storage used in the computation of planar convex hulls. In particular, we describe in-place and in situ algorithms for computing convex hulls. These are algorithms in which the input points are given as an array and the output, namely the vertices of the convex hull sorted in order of appearance on the hull, is returned in the same array. During the execution of the algorithm, additional working storage is kept to a minimum. In the case of in-place algorithms, the extra storage is kept in O(1) while in situ algorithms allow an extra memory of size O(log n). After execution of the algorithm, the array contains exactly the same points, but in a dierent order. In-place and in situ algorithms have several practical advantages over traditional algorithms. Primarily, because they use only a small additional amount of storage, they allow for the processing of larger data sets. Any algorithm that uses separate input and output arrays will, by necessity, require enough memory to store 2n points. In contrast, an in situ or in-place algorithm needs only enough memory to store n points plus O(log n) or O(1) working space, respectively. Related to this is the fact that in situ and in-place algorithms usually exhibit greater locality of reference, which makes them very practical for implementation on modern computer architectures with memory hierarchies. A nal advantage of in situ algorithms, especially in mission critical applications, is that they are less prone to failure since they do not require the allocation of large amounts of memory that may not be available at run time. We describe three planar convex hull algorithms. The rst is in-place, uses Grahams scan in combination with an in-place sorting algorithm, and runs in O(n log n) time. The second algorithm runs in O(n log h) time, is in situ and is based on an algorithm of Chan et al. [4]. The third (More Ultimate?) algorithm is based on an algorithm of Chan [3], runs in O(n log h) time and is in-place. The rst two algorithms are simple, implementable, and ecient in practice. To justify this claim, we have implemented both algorithms and made the source code freely available [25]. To the best of our knowledge, this paper is the rst to study the problem of computing convex hulls using in situ and in-place algorithms. This seems surprising, given the close relation between planar convex hulls and sorting, and the large body of literature on in situ sorting and merging algorithms [7,8,9,10,12,15,16,19,20,18,23,26,30,33,34,36]. The remainder of the paper is organized as follows: Sections 2, 3 and 4 describe our rst, second and third algorithms, respectively. Section 5 summarizes and concludes with open problems.
In this section, we present a simple in-place implementation of Grahams convex hull algorithm [13] or, more precisely, Andrews modication of Grahams algo-
rithm [1]. The algorithm requires the use of an in-place sorting algorithm. This can be any ecient in-place sorting algorithm (see, for example, [18,36]), so we refer to this algorithm simply as InPlace-Sort. Because this algorithm is probably the most practically relevant algorithm given in this paper, we begin by describing the most conceptually simple version of the algorithm, and then describe a slightly more involved version that improves the constants in the running time. 2.1 The Basic Algorithm
The upper convex hull of a point set S is the convex hull of S {(0, )}. The lower convex hull of a point set S is the convex hull of S {(0, )}. It is well-known that the convex hull of a point set is the intersection of its upper and lower convex hulls [29]. Grahams scan computes the upper (or lower) hull of an x-monotone chain incrementally, storing the partially computed hull on a stack. The addition of each new point involves removing zero or more points from the top of the stack and then pushing the new point onto the top of the stack. The following pseudo-code uses the InPlace-Sort algorithm and Grahams scan to compute the upper or lower hull of the point set S. The parameter d is used to determine whether the upper or lower hull is being computed. If d = 1, then InPlace-Sort sorts the points by increasing order of lexicographic (x, y)values and the upper hull is computed. If d = 1, then InPlace-Sort sorts the points by decreasing order and the lower hull is computed. The value of h corresponds to the number of elements on the stack. In the following, and in all remaining pseudo-code, S = S[0], . . . , S[n 1] is an array containing the input points. We use the C pointer notation S + i to denote the array S[i], . . . , S[n 1]. Graham-InPlace-Scan(S, n, d) 1: Graham-InPlace-Sort(S, n, d) 2: h 1 3: for i 1 . . . n 1 do 4: while h 2 and not right turn(S[h 2], S[h 1], S[i]) do 5: h h 1 { pop top element from the stack } 6: end while 7: swap S[i] S[h] 8: hh+1 9: end for 10: return h It is not hard to verify that when the algorithm returns in Line 10, the elements of S that appear on the upper (or lower) convex hull are stored in S[0], . . . , S[h 1]. In the case of an upper hull computation (d = 1), the hull vertices are sorted left-to-right (clockwise), while in the case of a lower hull computation (d = 1), the hull vertices are sorted right-to-left (also clockwise). To compute the convex hull of the point set S, we proceed as follows (refer to Fig. 1): First we make a call to Graham-InPlace-Scan to compute the
vertices of the upper hull of S and store them in clockwise order at positions S[0], . . . , S[h 1]. It follows that S[0] is the bottommost-leftmost point of S and that S[h1] is the topmost-rightmost point of S. We then use h1 swaps to bring S[0] to position S[h 1] while keeping the relative ordering of S[1], . . . S[h 1] unchanged. Finally, we make a call to Graham-InPlace-Scan to compute the lower convex hull of S[h 2], . . . , S[n 1] (which is also the lower convex hull of S). This stores the vertices of the lower convex hull in S[h 2], . . . , S[h + h 2] in clockwise order. The end result is that the convex hull of S is stored in S[0], . . . , S[h + h 2] in clockwise order.
A compute upper hull a S[1], . . . , S[h 1] move a b a S[h], . . . , S[n 1] compute lower hull S[0], . . . , S[h + h 2] output hull b S[h], . . . , S[n 1]
The following pseudo-code gives a more precise description of the algorithm. Graham-InPlace-Hull(S, n) 1: h Graham-InPlace-Scan(S, n, 1) 2: for i 0 . . . h 2 do 3: swap S[i] S[i + 1] 4: end for 5: h Graham-InPlace-Scan(S + h 2, n h + 2, 1) 6: return h + h 2 Each call to Graham-InPlace-Scan executes in O(n log n) time, and the loop in lines 24 takes O(h) time. Therefore, the total running time of the algorithm is O(n log n). The amount of extra storage used by InPlace-Sort is O(1), as is the storage used by both our procedures. Theorem 1 Algorithm Graham-InPlace-Hull computes the convex hull of a set of n points in O(n log n) time using O(1) additional memory.
2.2
The constants in the running time of Graham-InPlace-Hull can be improved by rst nding the extreme points a and b and using these points to partition the array into two parts, one that contains vertices that can only appear on the upper hull and one that contains vertices that can only appear on the lower hull. Fig. 2 gives a graphical description of this. In this way, each point (except a and b) takes part in only one call to Graham-InPlace-Scan. To further reduce the constants in the algorithm, one can implement InPlace-Sort with the in-place merge-sort algorithm of Katajainen et al. [18]. This algorithm requires only n log2 n + O(n) comparisons and 3 n log2 n + O(n) 2 swaps to sort n elements. Since Grahams scan performs only 2n h right-turn tests when computing the upper hull of n points having h points on the upper hull, the resulting algorithm performs at most 3nh right-turn tests (the extra n comes from the initial partitioning step). We call this algorithm Opt-GrahamInPlace-Hull.
A partition upper hull candidates compute upper hull a upper hull move a upper hull b a b lower hull candidates shift lower hull candidates compute lower hull convex hull output hull lower hull candidates
Theorem 2 Opt-Graham-InPlace-Hull computes the convex hull of n points in O(n log n) time using at most 3n h right turn tests, 3 n log2 n + O(n) 2 swaps, n log2 n + O(n) lexicographic comparisons and O(1) additional memory, where h is the number of vertices of the convex hull.
Finally, we note that if the array A is already sorted in lexicographic order then no lexicographic comparisons are necessary. One can use an in-place stable partitioning algorithm to partition A into the set of upper hull candidates and the set of lower hull candidates while preserving the sorted order within each set. There exists such an algorithm that runs in O(n) time and perform O(n) comparisons [19]. We call this algorithm Sorted-Graham-InPlace-Hull Theorem 3 Sorted-Graham-InPlace-Hull computes the convex hull of n points given in lexicographic order in O(n) time using O(n) right turn tests, O(n) swaps, no lexicographic comparisons and O(1) additional memory.
In this section, we show how to compute the upper (and symmetrically, lower) hull of S in O(n log h) time using an in situ algorithm, where h is the number of points of S that on the upper (respectively, lower) hull of S. We begin with a review of the O(n log h) time algorithm of Chan et al. [4]. To compute the upper hull of a point set S, we begin by arbitrarily grouping the elements of S into n/2 pairs. From these pairs, the pair with median slope s is found using a linear time median nding algorithm.1 We then nd a point p S such that the line through p with slope s has all points of S below it. Let q.x denote the x coordinate of the point q and let i denote the index of the element that is paired with S[i]. We now use p, and our grouping to partition the elements of S into three groups S 0 , S 1 , and S 2 as follows (see Fig. 3): 0 S if S[i].x p.x and (S[i], p) is not above S[i] S[i] S 1 if S[i].x > p.x and (S[i], p) is not above S[i] 2 S otherwise The algorithm then recursively computes the upper hull of S 0 {p} and S 1 {p} and outputs the concatenation of the two. For a discussion of correctness and a proof that this algorithm runs in O(n log h) time, see the original paper [4]. Now we turn to the problem of making this an in situ algorithm. The choice of median slope s ensures that S 0 3n/4 and S 1 3n/4, so the algorithm uses only O(log n) levels of recursion. Our strategy is to implement each level using O(1) local variables and one call to a median-nding routine that uses O(log n) additional memory. For simplicity, assume n is even. The case when n is odd is easily handled by processing an extra unpaired element after all the paired elements have been processed. To pair o elements, we pair S[i] with S[i + 1] if i is even and with S[i 1] if i is odd. Several in situ linear time median nding algorithms exist (see, for example, Horowitz et al. [14, Section 3.6] or Lai and Wood [22]), so
1
Bhattacharya and Sen [2] and Wenger [35] have both noted that median nding can be replaced by choosing a random pair of elements. The expected running time of the resulting algorithm is O(n log h).
S[i] S[i]
S[i] S 0
S[i] S[i]
S[i] S 1
p S[i]
p S[i]
S[i]
S[i] S
2
S[i]
S[i] S 2
nding the pair (S[i], S[i + 1]) with median slope can be done with one of these. The tricky part of the implementation is the partitioning into sets S 0 , S 1 and S 2 . The diculty lies in the fact that the elements are grouped into pairs, but the two elements of the same pair may belong to dierent sets S i and S j . We rst note that we can compute the sizes n0 , n1 and n2 of these sets in linear time without diculty by scanning S. Conceptually, we partition S into three les, f0 , f1 and f2 that contain pairs of points in S. The le f0 contains the elements S[0], . . . , S[2 n0 /2 1]. The le f1 contains the elements S[2 n0 /2 ], . . . , S[2 (n0 +n1 )/2 1]. The le f2 contains the elements S[2 (n0 + n1 )/2 ], . . . , S[n]. It is important to note that these les are only abstractions. Each le fi is implemented using two integer values ri and i . The value of ri is initialized to the index of the rst record in the le. The value of i is initialized to ri + ki where ki is the number of elements in fi . A Read operation on fi returns the pair (S[ri ], S[ri+1 ]) and increases the value of ri by 2. We say that fi is empty if ri i . These les are used in conjunction with a stack A that stores pairs of points. The stack and les serve two purposes: (1) when there is no data on the stack we read a pair from one of the les and store it on the stack, and (2) when we are about to overwrite an element from a pair that has not yet been placed on the stack, we read the pair from the le and save it on the stack. In this way no element is ever overwritten without rst being saved on the stack, and the initial pairing of elements is preserved.
These ideas are made more concrete by the following pseudo-code, which places the elements of S 0 into array locations S[0], . . . , S[n0 1], the elements of S 1 into array locations S[n0 ], . . . , S[n0 + n1 1], and the elements of S 2 into array locations S[n0 +n1 ], . . . , S[n1]. The algorithm repeatedly processes pairs (a, b) of elements by determining which of the three sets a and b belong to and then placing a and b in their correct locations. CSY-Partition(S, n, n0 , n1 ) 1: i0 0 2: i1 n0 3: i2 n0 + n1 4: m 0 5: while m > 0 or one of f0 , f1 , f2 is not empty do 6: if m = 0 then 7: A[m] ReadFromFile() 8: mm+1 9: end if 10: mm1 11: P A[m] { process this pair } 12: for both q P do 13: S j Group(q, P ) 14: Place(q, ij ) 15: ij ij + 1 16: end for 17: end while The ReadFromFile function simply reads a pair from one of the non-empty les and returns it. The Group(q, P ) returns (a pointer to) the group of point q in the pair P . The Place(q, k) function places the point q at index k in S, after ensuring that the overwritten element has been read and placed on the stack. Place(q, k) 1: for i 0, 1, 2 do 2: if k ri and k < i then 3: { S[k] belongs to fi and has not yet been read } 4: Read(a, b) from fi 5: A[m] (a, b) 6: mm+1 7: end if 8: end for 9: S[k] q To show that this partitioning step is correct, we make 2 observations. (1) Exactly n/2 pairs of elements are read and processed since the le abstraction ensures that no pair is read more than once and the algorithm does not terminate until all les are empty. (2) The code in Place ensures that any pair is read and placed on the stack A before an element of the pair is overwritten. Therefore,
all of the original n/2 pairs of elements are processed and each element is placed into one of S 0 , S 1 or S 2 . Since the algorithm uses a stack A that may grow without bound, it is not obvious that the partitioning algorithms additional memory is of a constant size. To prove that A does not grow without bound note that overwriting ki elements of fi causes at most ki /2 read operations. Each iteration of the outer loop places one pair of elements, and each read operation reads one pair of elements. Therefore, the total number of read operations performed after k iterations is at most k + 3. However, each iteration removes 1 pair of elements from the stack A, so the total number of pairs on the stack after k iterations is not more than 3. Since this holds for any value of k, the stack A never holds more than 3 pairs of elements. Fig. 4 recaps the algorithm for computing the upper hull of S. First the algorithm partitions S into the sets S 0 , S 1 and S 2 . It then recurses on the set S 0 . After the recursive call, the convex hull of S 0 is stored at the beginning of the array S, and the last element of this hull is the point p that was used for partitioning. The algorithm then shifts S 1 leftward so that it is adjacent to p and recurses on S 1 {p}. The end result is the upper hull of S being stored consecutively and in clockwise order at the beginning of the array S.
Using the technique from Section 2 (Figures 1 and 2), this upper hull algorithm can be made in to a convex hull algorithm with the same running time and memory requirements. Theorem 4 Algorithm CSY-InSitu-Hull computes the convex hull of n points in O(n log h) time using O(log n) additional storage, where h is the number of vertices of the convex hull.
Next, we give an O(n log h) time in-place planar convex hull algorithm. We begin with a review of Chans O(n log h) time algorithm [3], which is essentially a speedup of Jarvis March [17]. Chans algorithm runs in rounds. During ith round the algorithm nds the i rst gi = 22 points on the convex hull. Once gi h the rounds end as the algorithm detects that it has found all points on the convex hull. During round i, the algorithm partitions the input points into n/gi groups of size gi and computes the convex hull of each group. The vertices on the convex hull are output in clockwise order beginning with the leftmost vertex. Each successive vertex is obtained by nding tangents from the previous vertex to each of the n/gi convex hulls. The next vertex is determined, as in Jarvis March, by choosing the vertex having largest polar angle with respect to the previously found vertex as origin. In the case where the largest polar angle is not unique, ties are broken by taking the farthest vertex from the previously found vertex. Finding a tangent to an individual convex hull can be done in O(log gi ) time if the vertices of the convex hull are stored in an array in clockwise order [5,29]. There are n/gi tangent nding operations per iteration and gi iterations in round i. Therefore, round i takes O(n log gi ) = O(n2i ) time. Since there are at most log log h log log h rounds, the total cost of Chans algorithm is O(n2i ) = i=1 O(n log h). Next we show how to implement each round using only O(1) additional storage. Assume for the sake of simplicity that n is a multiple of gi . For the grouping step, we build n/gi groups of size gi by taking groups of consecutive elements in S and computing their convex hulls using Graham-InPlace-Hull. Two questions now arise: (1) Once we start the tangent-nding steps, where do we put the convex hull vertices as we nd them? (2) In order to nd a tangent from a point to a group in O(log gi ) time we need to know the size of the convex hull of the group. How can we keep track of all these sizes using only O(1) extra memory? To answer the rst question, we store convex hull vertices at the beginning of the array S in the order that we nd them. That is, when we nd the kth vertex on the convex hull, we swap it with S[k 1]. We say that a group G is dirty if one of its members has been found to be on the convex hull of S. A group that is not dirty is clean. If a group G is clean, then we can use binary search to nd a tangent to G in O(log gi ) time, otherwise we have to use linear search which takes O(gi ) time. Since after k iterations, the algorithm stores the rst
k hull vertices in locations S[0], . . . S[k 1], the rst group consists of elements S[k], . . . , S[gi 1] and is always considered dirty. To keep track of which other groups are dirty, we mark them by reordering the rst two points of the group. In a clean group, the points are stored in lexicographic order. In a dirty group, we store them in reverse lexicographic order. This allows us to test in constant time whether a group is clean or dirty. To keep track of the size of the convex hull of each clean group without storing the size explicity we use another reordering trick. Let G[0], . . . , G[gi 1] denote the elements of a clean group G and let < denote lexicographic comparison of (x, y) values. We say that the sign of G[j] is + if G[j] < G[j+1], and otherwise. If the convex hull of G contains h vertices, then it follows that the rst elements G[0], . . . , G[h 2] have signs that form a sequence of 1 or more +s followed by 0 or more s. Furthermore, the elements G[h], . . . , G[gi 1] can be reordered so that the remainder of the signs form an alternating sequence. When we do this, a group element G[j], 0 < j < gi 1, j = h 1 is on the convex hull of G if and only if G[j 1], G[j], G[j + 1] do not have signs that alternate. As for G[0], G[gi 1] and G[h 1] we know that G[0] is always on the convex hull of G. The point G[gi 1] is on the convex hull of G if and only if G[gi 2] is on the convex hull of G and the triangle G[gi 2], G[gi 1], G[0] is oriented clockwise.2 The point G[h 1] is on the convex hull of G if and only if G[h 2] is on the convex hull of G and the triangle G[h 2], G[h 1], G[0] is oriented clockwise. Therefore, for any index 0 j < gi , we can test if G[j] is on the convex hull of G in constant time. Using this in conjunction with binary search, we can compute the number of vertices on the convex hull of G in O(log gi ) time. Thus, we can compute the size of the convex hull of G and nd a tangent in O(log gi ) time, as required. We have provided all the tools for an in-place implementation of Chans algorithm. Except for the extra cost of nding tangents in dirty groups, the running time of this implementation is asymptotically the same as that of the original algorithm, so we need only bound this extra cost of nding tangents in dirty groups. During one step of round i, we nd tangents of at most gi dirty groups, at a cost of O(gi ) per group, and there are gi steps in round i. Therefore, 3 the total cost of searching dirty groups during round i is O(gi ) O(n) for all 1/3 3 gi n . Therefore, the total cost of round i is O(gi + n log gi ) O(n log gi ) for any h < n1/3 . Since we can abort the algorithm when gi n1/3 and use Graham-InPlace-Hull, the overall running time of the algorithm is again O(n log h). Theorem 5 The above algorithm, Chan-InPlace-Hull, computes the convex hull of n points in O(n log h) time using O(1) additional storage, where h is the number of vertices of the convex hull. The constants in Chan-InPlace-Hull can be improved using the following trick that is mentioned by Chan [3]. When round i terminates without nding the entire convex hull, the gi convex hull points that were computed should not
2
We use the convention that three collinear points are not oriented clockwise.
be discarded. Instead, the grouping in round i+1 is done on the remaining ngi points, thus eliminating the need to recompute the rst gi convex hull vertices. This optimization works perfectly when applied to Chan-InPlace-Hull since the rst gi convex hull points are already stored at locations S[0], . . . , S[gi 1].
Conclusions
We have given three algorithms for computing the convex hull of a planar point set. The rst algorithm is in-place and runs in O(n log n) time. The second algorithm is in situ and runs in O(n log h) time. The third algorithm is in-place and and runs in O(n log h) time. The rst two algorithms are reasonably simple and implementable. In order to facilitate comparisons with other convex hull implementations, our source code is available for download [25]. This paper came to be when two separate groups of researchers (authors 13 and authors 46) discovered they were both working on in-place computational geometry algorithms and decided to merge their results. Some of these results have been omitted due to space constraints. These include in-place or in situ implementations of Eddys algorithm (also known as quickhull) [11], Kirkpatrick and Seidels algorithm [21], Seidels randomized linear programming algorithm [31] and Megiddos deterministic linear programming algorithm [24]. The ideas presented in this paper also apply to other problems. The maximal elements problem is that of determining all elements S[i] such that S[j] < S[i] for all 0 j < n. An algorithm almost identical to Grahams scan can be used to solve the maximal elements problems in O(n log n) time, and this can easily be implemented in-place. Furthermore, an in-place algorithm almost identical to that in Section 4 can be used to solve the maximal elements problem in O(n log h) time, where h is the number of maximal elements. The question of in situ and in-place algorithms for maximal elements and convex hulls in dimensions d 3 is still open. In order for this question to make sense, we ask only that the algorithm identify which input points are maximal or on the convex hull. Testing whether a given point is maximal can be done in O(dn) time using the denition of maximality. Testing whether a single point is on the convex hull is a d 1 dimensional linear programming problem that can be solved in-place in O(d!n) expected time using Seidels algorithm [31]. Thus, the maximal elements problem can be solved in O(dn2 ) time and the convex hull problem can be solved in O(d!n2 ) time using in-place algorithms. Are there algorithms with reduced dependence on n? More generally, one might ask what other computational geometry problems admit in-place or in situ algorithms. Some problems that immediately come to mind are those of computing k-piercings of sets, nding maximum cliques in intersection graphs, computing largest empty disks, computing smallest enclosing disks, and nding ham-sandwich cuts.
References
1. A. M. Andrew. Another ecient algorithm for convex hulls in two dimensions. Information Processing Letters, 9:216219, 1979. Corrigendum, Information Processing Letters, 10:168, 1980. 2. B. K. Bhattacharya and S. Sen. On a simple, practical, optimal, output-sensitive randomized planar convex hull algorithm. Journal of Algorithms, 25(1):177193, 1997. 3. T. Chan. Optimal output-sensitive convex hull algorithms in two and three dimensions. Discrete & Computational Geometry, 16:361368, 1996. 4. T. Chan, J. Snoeyink, and C. K. Yap. Primal dividing and dual pruning: Output-sensitive construction of four-dimensional polytopes and three-dimensional Voronoi diagrams. Discrete & Computational Geometry, 18:433454, 1997. 5. B. Chazelle and D. P. Dobkin. Intersection of convex objects in 2 and 3 dimensions. Journal of the ACM, 34:127, 1987. 6. K. L. Clarkson and P. W. Shor. Applications of random sampling in computational geometry, II. Discrete & Computational Geometry, 4(1):387421, 1988. 7. E. W. Dijkstra. Smoothsort, an alternative for sorting in situ. Science of Computer Programming, 1(3):223233, 1982. 8. E. W. Dijkstra and A. J. M. van Gasteren. An introduction to three algorithms for sorting in situ. Information Processing Letters, 15(3):129134, 1982. 9. S. Dvorak and B. Durian. Stable linear time sublinear space merging. The Computer Journal, 30(4):372375, 1987. 10. S. Dvorak and B. Durian. Unstable linear time O(1) space merging. The Computer Journal, 31(3):279282, 1988. 11. W. Eddy. A new convex hull algorithm for planar sets. ACM Transactions on Mathematical Software, 3(4):398403, 1977. 12. R. W. Floyd. Algorithm 245, Treesort 3. Communications of the ACM, 7:401, 1964. 13. R. L. Graham. An ecient algorithm for determining the convex hull of a nite planar set. Information Processing Letters, 1:132133, 1972. 14. E. Horowitz, S. Sahni, and S. Rajasekaran. Computer Algorithms. Computer Science Press, 1998. 15. B.-C. Huang and M. A. Langston. Practical in-place merging. Communications of the ACM, 31(3):348352, 1988. 16. B.-C. Huang and M. A. Langston. Fast stable merging and sorting in constant extra space. The Computer Journal, 35(6):643650, 1992. 17. A. Jarvis. On the identication of the convex hull of a nite set of points in the plane. Information Processing Letters, 2:1821, 1973. 18. J. Katajainen, T. Pasanen, and J. Teuhola. Practical in-place mergesort. Nordic Journal of Computing, 3:2740, 1996. 19. J. Katajainen and T. A. Pasanen. Stable minimum space partitioning in linear time. BIT, 32(4):580585, 1992. 20. J. Katajainen and T. A. Pasanen. In-place sorting with fewer moves. Information Processing Letters, 70(1):3137, 1999. 21. D. G. Kirkpatrick and R. Seidel. The ultimate planar convex hull algorithm? SIAM Journal on Computing, 15(1):287299, 1986. 22. T. W. Lai and D. Wood. Implicit selection. In Proceedings of the 1st Scandinavian Workshop on Algorithm Theory, volume 318 of Lecture Notes in Computer Science, pages 1423. Springer-Verlag, 1988.
23. H. Mannila and E. Ukkonen. A simple linear-time algorithm for in situ merging. Information Processing Letters, 18(4):203208, 1984. 24. N. Megiddo. Linear programming in linear time when the dimension is xed. Journal of the ACM, 31(1):114127, 1984. 25. P. Morin. insitu.tgz. Available online at http://cgm.cs.mcgill.ca/~morin/, 2001. 26. J. I. Munro, V. Raman, and J. S. Salowe. Stable in situ sorting and minimum data movement. BIT, 30(2):220234, 1990. 27. F. P. Preparata. An optimal real time algorithm for planar convex hulls. Communications of the ACM, 22:402405, 1979. 28. F. P. Preparata and S. J. Hong. Convex hulls of nite point sets in two and three dimensions. Communications of the ACM, 2(20):8793, 1977. 29. F. P. Preparata and M. I. Shamos. Computational Geometry. Springer-Verlag, 1985. 30. J. Salowe and W. Steiger. Simplied stable merging tasks. Journal of Algorithms, 8(4):557571, 1987. 31. R. Seidel. Small-dimensional linear programming and convex hulls made easy. Discrete & Computational Geometry, 6:423434, 1991. 32. M. I. Shamos. Computational Geometry. PhD thesis, Yale University, 1978. 33. H. W. Six and L. Wegner. Sorting a random access le in situ. The Computer Journal, 27(3):270275, 1984. 34. A. Symvonis. Optimal stable merging. The Computer Journal, 38(8):681690, 1995. 35. R. Wenger. Randomized quick hull. Algorithmica, 17:322329, 1997. 36. J. W. J. Williams. Algorithm 232, Heapsort. Communications of the ACM, 7:347 348, 1964.