Convex Hull
A convex hull of a shape is defined as:
In mathematics, the convex hull or convex envelope for a set of points X in a real vector space V is the minimal convex set containing X (Wikipedia)
Wikipedia visualizes it nicely using a rubber band analogy, and there are some good algorithms to compute it.
Concave Hull
A concave hull is visualized using the red line in the image below (the blue line visualizes the convex hull). Intuitively, it is a polygon which embraces all the points, but has less (minimal?) area compared to the convex hull. As a result, the polygon's boundary length is longer.
A concave hull may be the solution for some real-world problems (e.g. finding the reasonable boundary of a city).
enter image description here
Is there a proper definition, algorithm and practical solution for the notion of a Concave Hull?
The Grass Wiki has some descriptions and images, and there is a commercial solution in concavehull.com.
14 Answers 14
As scw points out, you want an implementation of α-shapes.
Alpha shapes can be considered a generalisation of the convex hull. They were first described in 1981 in:
Edelsbrunner, H.; Kirkpatrick, D.; Seidel, R.; , "On the shape of a set of points in the plane," Information Theory, IEEE Transactions on , vol.29, no.4, pp. 551- 559, Jul 1983
Open source implementations exist for the environments you are interested in:
PostGIS
If you are using the PostGIS stack, pgRouting's optional Driving Distance extension exposes an alpha shape implementation. I'm not sure if you can vary the alpha parameter, however.
Usage is below:
SELECT the_geom AS alpha_shape
FROM
points_as_polygon(
'SELECT id, ST_X(your_geom) AS x, ST_Y(your_geom) AS y FROM your_table');
Python
There are probably many Python modules you could use. I have heard good things about CGAL, a C++ computational geometry library. Python wrappers exist for parts of CGAL, including exposing CGAL's alpha shape implementation to Python.
Be aware that parts of CGAL are licensed under the QPL, which means that if you distribute your program, linked to CGAL, you may need to release it under the QPL. It is fine to keep your code proprietary if you do not redistribute your program code or binaries.
This seems to be a specific application of alpha shapes, which are from my reading a more general form of this problem. R has the alphahull module, which has excellent documentation on computing alpha shapes. Also check this detailed background on alpha shapes. If you only want to compute convex/concave hulls, check out lasboundary, part of lastools, it scales well and can handle millions of input points.
Finally, this interesting application of alpha shapes by Flickr made the rounds a while ago, showing their utility for aggregating user generated point content:
alpha hull of texas from flickr
There is an implementation of ST_ConcaveHull in PostGIS trunk. http://postgis.net/docs/ST_ConcaveHull.html
I created a highly-efficient tool, called lasboundary (1,2), that computes a concave hull for LIDAR in LAS/LAZ/SHP/ASCII format and stores the result as a vector boundary polygon in ESRI Shapefile format or a geo-referenced KML file.
Here is an example run:
C:\lastools\bin>lasboundary -i SerpentMound.las -o SerpentMound_boundary.shp
reading 3265110 points and computing convex hull for 3265110 points
growing inward towards concave hull (with concavity = 50)
outputting the concave hull
concave hull has 1639 points
Some result pictures are here.
Here is an R function that implements the Alpha Hull model. The output is an sp polygon object. Please see the example in the header. It requires the sp, alphahull and maptools packages.
**Update (01-15-2018) There have been numerous changes to the resulting objects produced by the alphahull package. As such, I needed to rewrite the function. I added a convexHull function to the spatialEco package. However, due to licensing restrictions in the alphahull package this function is only available in the development version on GitHub. The development version can be installed using:
library(devtools)
install_github("jeffreyevans/spatialEco")
Here is an example of the functions usage
library(sp)
library(spatialEco)
data(meuse)
coordinates(meuse) = ~x+y
a <- convexHull(meuse, alpha=100000)
plot(a)
points(meuse, pch=19)
Convert the resulting SpatialLinesDataFrame to SpatialPolygonsDataFrame
library(sf)
a <- sf::st_as_sf(a)
a <- sf::st_polygonize(a)
class( a <- as(a, "Spatial") )
plot(a)
Test multiple alpha values
par(mfcol=c(2,2))
for (a in c(500, 1500, 5000, 100000)) {
ch <- convexHull(meuse, alpha = a)
plot(ch)
points(meuse, pch=19)
title( paste0("alpha=", a))
}
About R implementation Alpha-Shapes, there's an article about "Converting Alpha-Shapes into SP Objects"
It's based on alphahull, sp and spgrass6 http://casoilresource.lawr.ucdavis.edu/drupal/node/919
JTS (https://github.com/locationtech/jts) provides a Convex Hull implementation. Martin Davies also mentioned having an Alpha Shape algorithm in the works so you might want to check the SVN repository to see if it is in yet if that's what you want.
Speaking about JTS, you can use Geoscript for manipulating JTS library. http://geoscriptblog.blogspot.com/2010/06/unwrapped-jts-with-python.html for an article about convex hull. GeoScript implementations are available in JavaScript, Python, Scala, and Groovy. The official website : http://geoscript.org
Here's a program written in C that computes convex hulls, alpha shapes, Delauney triangluations and Voronoi volumes:
- Hull - Ken Clarkson (2002)
Another convex hull algorithm with C and Java implementations is here:
- Convex Hull (2D) - Computational Geometry in C, Joseph O'Rourke (1998)
-
The second link appears to be broken.2023年04月17日 08:41:55 +00:00Commented Apr 17, 2023 at 8:41
QGIS does have Concave hull algorithm.
Parameters
Input point layer [vector: point]
put parameter description here
Threshold (0-1, where 1 is equivalent with Convex Hull) [number]
put parameter description here
Default: 0.3
Allow holes [boolean]
put parameter description here
Default: True
Split multipart geometry into singleparts geometries [boolean]
Default: False
Outputs Concave hull [vector]
put output description here
Console usage
processing.runalg('qgis:concavehull', input, alpha, holes, no_multigeometry, output)
Also, Esri has a tool Minimum Bounding Geometry (Data Management)
Which allows you to choose the geometry type, which includes convex hull
To help with the "proper definition" part of your question; you may have looked at https://en.wikipedia.org/wiki/Convex_hull and gotten what could well be considered a "proper" definition, but found it lacking, so perhaps a more "useful" definition is:
For every point inside a convex hull, a straight line to any point not within the hull will only intersect the hull once.
This is useful because given a point you can construct a line through it and test for that constructed line intersecting segments of the hull.
- No intersection the point is not in the hull.
- One intersection the point is on the hull.
- Two intersections the point is within the hull
- A straight line cannot intersect a convex hull more than twice
-
3the op is asking about concave hulls, and not convex hullswinwaed– winwaed2015年06月29日 02:31:58 +00:00Commented Jun 29, 2015 at 2:31
JTS now has a Concave Hull implementation. This is based on erosion of the Delaunay triangulation of the input points. It supports creating holes, and provides both absolute and scale-free length parameters.
The algorithm has also been ported to GEOS, and exposed in the next version (3.3) of PostGIS.
Assume that you begin with black points drawn on a white plane.
One approach would be to...
find the centroid of the point cloud.
convert to polar coordinates.
sort the points based on the angle-part of the polar coordinate.
draw a polygonal chain with straight line-segments from dot-to-dot.
fill the inside of the region enclosed by the polygonal chain with pure black.
observe that the result has a lot of spikes, like a pincushion or a Japanese Shuriken
Let
maxdist
be the the maximum distance from any point in the point cloud to the center of the point cloud.Grow the the selection by
0.25*maxdist
("grow selection" comes from the open source GNU image manipulation program)shrink the the selection by
0.25*maxdist
("shink selection" comes from the open source GNU image manipulation program )
Note that the shink function is not the inverse of the grow function. For any shape x, shrink(grow(x)) ≠ x.
Growing the selection removes any valleys or spikes on the surface of the shape, but makes the shape too large.
Shrinking the result from the grow operation will make the shape the correct size again.
Explore related questions
See similar questions with these tags.