Geohash a lng/lat coordinate using the hilbert curve.
Project description
geohash-hilbert
Geohash a lng/lat coordinate using hilbert space filling curves.
In [1]: import geohash_hilbert as ghh
In [2]: ghh.encode(6.957036, 50.941291)
Out[2]: 'Z7fe2GaIVO'
In [3]: ghh.decode('Z7fe2GaIVO')
Out[3]: (6.957036126405001, 50.941291032359004)
In [4]: ghh.decode_exactly('Z7fe2GaIVO')
Out[4]:
(6.957036126405001, 50.941291032359004, # position
1.6763806343078613e-07, 8.381903171539307e-08) # errors
In [5]: ghh.encode?
Signature: ghh.encode(lng, lat, precision=10, bits_per_char=6)
Docstring:
Encode a lng/lat position as a geohash using a hilbert curve
This function encodes a lng/lat coordinate to a geohash of length `precision`
on a corresponding a hilbert curve. Each character encodes `bits_per_char` bits
per character (allowed are 2, 4 and 6 bits [default 6]). Hence, the geohash encodes
the lng/lat coordinate using `precision` * `bits_per_char` bits. The number of
bits devided by 2 give the level of the used hilbert curve, e.g. precision=10, bits_per_char=6
(default values) use 60 bit and a level 30 hilbert curve to map the globe.
Parameters:
lng: float Longitude; between -180.0 and 180.0; WGS 84
lat: float Latitude; between -90.0 and 90.0; WGS 84
precision: int The number of characters in a geohash
bits_per_char: int The number of bits per coding character
Returns:
str: geohash for lng/lat of length `precision`
File: .../geohash_hilbert/_hilbert.py
Type: function
In [7]: ghh.decode?
Signature: ghh.decode(code, bits_per_char=6)
Docstring:
Decode a geohash on a hilbert curve as a lng/lat position
Decodes the geohash `code` as a lng/lat position. It assumes, that
the length of `code` corresponds to the precision! And that each character
in `code` encodes `bits_per_char` bits. Do not mix geohashes with different
`bits_per_char`!
Parameters:
code: str The geohash to decode.
bits_per_char: int The number of bits per coding character
Returns:
Tuple[float, float]: (lng, lat) coordinate for the geohash.
File: .../geohash_hilbert/_hilbert.py
Type: function
In [8]: ghh.decode_exactly?
Signature: ghh.decode_exactly(code, bits_per_char=6)
Docstring:
Decode a geohash on a hilbert curve as a lng/lat position with error-margins
Decodes the geohash `code` as a lng/lat position with error-margins. It assumes,
that the length of `code` corresponds to the precision! And that each character
in `code` encodes `bits_per_char` bits. Do not mix geohashes with different
`bits_per_char`!
Parameters:
code: str The geohash to decode.
bits_per_char: int The number of bits per coding character
Returns:
Tuple[float, float, float, float]: (lng, lat, lng-error, lat-error) coordinate for the geohash.
File: .../geohash_hilbert/_hilbert.py
Type: function
Compare to original geohash
This package is similar to the geohash or the geohash2 package, as it also provides a mechanism to encode (and decode) a longitude/latitude position (WGS 84) to a one dimensional geohash. But, where the former use z-order space filling curves, this package uses hilbert curves. (The kernel for this package was adapted from wiki).
Note: The parameter (and returns) order changed from lat/lng in geohash
to lng/lat. Apart from that this package is a drop-in replacement for the original geohash
.
Further, the string representation is changed (and modifieable) to compensate for the special requirements of the implementation: geohash
uses a modified base32 representation, i.e. every character in the geohash encodes 5 bits. Even bits encode longitude and odd bits encode latitude. Every two full bits encode for one level of the z-order curve, e.g. the default precision of 12 use 12*5 = 60bit
to encode one latitude / longitude position using a level 30 z-order curve. The implementation also allows for 'half'-levels, e.g. precision 11 use 11*5 = 55bit
corresponds to a level 27.5 z-order curve.
Geohash representation details
This implementation of the hilbert curve allows only full levels, hence we have support for base4 (2bit), base16 (4bit) and a custom base64 (6bit, the default) geohash representations. All keep the same ordering as their integer value by lexicographical order:
- base4: each character is in
'0123'
- base16: each character is in
'0123456789abcdef'
- base64: each character is in
'0123456789@ABCDEFGHIJKLMNOPQRSTUVWXYZ_abcdefghijklmnopqrstuvwxyz'
Note: Do not mix geohashes from the original geohash
and this, and do not mix base4, base16 and base64 geohash representations. Decide for one representation and then stick to it.
The different encodings also give a more fine grain control of the actual encoding precision and the geohash size (maximum lng/lat error around equator):
lvl | bits | error | base4 | base16 | base64
-------------------------------------------------------------
0 | 0 | 20015.087 km | prec 0 | prec 0 | prec 0
1 | 2 | 10007.543 km | prec 1 | |
2 | 4 | 5003.772 km | prec 2 | prec 1 |
3 | 6 | 2501.886 km | prec 3 | | prec 1
4 | 8 | 1250.943 km | prec 4 | prec 2 |
5 | 10 | 625.471 km | prec 5 | |
6 | 12 | 312.736 km | prec 6 | prec 3 | prec 2
7 | 14 | 156.368 km | prec 7 | |
8 | 16 | 78.184 km | prec 8 | prec 4 |
9 | 18 | 39.092 km | prec 9 | | prec 3
10 | 20 | 19.546 km | prec 10 | prec 5 |
11 | 22 | 9772.992 m | prec 11 | |
12 | 24 | 4886.496 m | prec 12 | prec 6 | prec 4
13 | 26 | 2443.248 m | prec 13 | |
14 | 28 | 1221.624 m | prec 14 | prec 7 |
15 | 30 | 610.812 m | prec 15 | | prec 5
16 | 32 | 305.406 m | prec 16 | prec 8 |
17 | 34 | 152.703 m | prec 17 | |
18 | 36 | 76.351 m | prec 18 | prec 9 | prec 6
19 | 38 | 38.176 m | prec 19 | |
20 | 40 | 19.088 m | prec 20 | prec 10 |
21 | 42 | 954.394 cm | prec 21 | | prec 7
22 | 44 | 477.197 cm | prec 22 | prec 11 |
23 | 46 | 238.598 cm | prec 23 | |
24 | 48 | 119.299 cm | prec 24 | prec 12 | prec 8
25 | 50 | 59.650 cm | prec 25 | |
26 | 52 | 29.825 cm | prec 26 | prec 13 |
27 | 54 | 14.912 cm | prec 27 | | prec 9
28 | 56 | 7.456 cm | prec 28 | prec 14 |
29 | 58 | 3.728 cm | prec 29 | |
30 | 60 | 1.864 cm | prec 30 | prec 15 | prec 10
31 | 62 | 0.932 cm | prec 31 | |
32 | 64 | 0.466 cm | prec 32 | prec 16 |
-------------------------------------------------------------
Further features
If cython is available during install, the cython kernel extension will be installed and used for geohash computations with 64bit or less (timings for MBP 2016, 2.6 GHz Intel Core i7, Python 3.6.2, Cython 0.26.1):
In [1]: import geohash_hilbert as ghh
# Without cython ...
In [2]: ghh._hilbert.CYTHON_AVAILABLE
Out[2]: False
In [3]: %timeit ghh.encode(6.957036, 50.941291, precision=10)
39.4 µs ± 614 ns per loop (mean ± std. dev. of 7 runs, 10000 loops each)
In [4]: %timeit ghh.encode(6.957036, 50.941291, precision=11)
43.4 µs ± 421 ns per loop (mean ± std. dev. of 7 runs, 10000 loops each)
In [1]: import geohash_hilbert as ghh
# With cython ...
In [2]: ghh._hilbert.CYTHON_AVAILABLE
Out[2]: True
# almost 6x faster
In [3]: %timeit ghh.encode(6.957036, 50.941291, precision=10)
6.72 µs ± 57.4 ns per loop (mean ± std. dev. of 7 runs, 100000 loops each)
# more than 64bit will be computed with pure python function.
In [4]: %timeit ghh.encode(6.957036, 50.941291, precision=11)
43.4 µs ± 375 ns per loop (mean ± std. dev. of 7 runs, 10000 loops each)
Get the actual rectangle that is encoded by a geohash, i.e. position +- errors:
# returns a geojson Feature encoding the rectangle as a Polygon
In [9]: ghh.rectangle('Z7fe2G')
Out[9]:
{'bbox': (6.955718994140625,
50.94085693359375,
6.95709228515625,
50.94154357910156),
'geometry': {'coordinates': [[(6.955718994140625, 50.94085693359375),
(6.955718994140625, 50.94154357910156),
(6.95709228515625, 50.94154357910156),
(6.95709228515625, 50.94085693359375),
(6.955718994140625, 50.94085693359375)]],
'type': 'Polygon'},
'properties': {'bits_per_char': 6,
'code': 'Z7fe2G',
'lat': 50.941200256347656,
'lat_err': 0.00034332275390625,
'lng': 6.9564056396484375,
'lng_err': 0.0006866455078125},
'type': 'Feature'}
Get the neighbouring geohashes:
In [10]: ghh.neighbours('Z7fe2G')
Out[10]:
{'east': 'Z7fe2T',
'north': 'Z7fe2H',
'north-east': 'Z7fe2S',
'north-west': 'Z7fe2I',
'south': 'Z7fe2B',
'south-east': 'Z7fe2A',
'south-west': 'Z7fe2E',
'west': 'Z7fe2F'}
Plot the Hilbert curve:
# returns a geojson Feature encoding the Hilbert curve as a LineString
In [11]: ghh.hilbert_curve(1) # this is a level 3 Hilbert curve:
# 1 char * 6 bits/char = 6 bits => level 3
Out[11]:
{'geometry': {'coordinates': [(-157.5, -78.75),
(-157.5, -56.25), (-112.5, -56.25), (-112.5, -78.75), (-67.5, -78.75), (-22.5, -78.75),
(-22.5, -56.25), (-67.5, -56.25), (-67.5, -33.75), (-22.5, -33.75), (-22.5, -11.25),
(-67.5, -11.25), (-112.5, -11.25), (-112.5, -33.75), (-157.5, -33.75), (-157.5, -11.25),
(-157.5, 11.25), (-112.5, 11.25), (-112.5, 33.75), (-157.5, 33.75), (-157.5, 56.25),
(-157.5, 78.75), (-112.5, 78.75), (-112.5, 56.25), (-67.5, 56.25), (-67.5, 78.75),
(-22.5, 78.75), (-22.5, 56.25), (-22.5, 33.75), (-67.5, 33.75), (-67.5, 11.25),
(-22.5, 11.25), (22.5, 11.25), (67.5, 11.25), (67.5, 33.75), (22.5, 33.75), (22.5, 56.25),
(22.5, 78.75), (67.5, 78.75), (67.5, 56.25), (112.5, 56.25), (112.5, 78.75), (157.5, 78.75),
(157.5, 56.25), (157.5, 33.75), (112.5, 33.75), (112.5, 11.25), (157.5, 11.25), (157.5, -11.25),
(157.5, -33.75), (112.5, -33.75), (112.5, -11.25), (67.5, -11.25), (22.5, -11.25),
(22.5, -33.75), (67.5, -33.75), (67.5, -56.25), (22.5, -56.25), (22.5, -78.75),
(67.5, -78.75), (112.5, -78.75), (112.5, -56.25), (157.5, -56.25), (157.5, -78.75)],
'type': 'LineString'},
'properties': {},
'type': 'Feature'}
Project details
Release history Release notifications | RSS feed
Download files
Download the file for your platform. If you're not sure which to choose, learn more about installing packages.
Source Distribution
File details
Details for the file geohash-hilbert-1.2.1.tar.gz
.
File metadata
- Download URL: geohash-hilbert-1.2.1.tar.gz
- Upload date:
- Size: 2.2 MB
- Tags: Source
- Uploaded using Trusted Publishing? No
- Uploaded via: poetry/0.11.4 CPython/3.6.3 Linux/4.14.12-041412-generic
File hashes
Algorithm | Hash digest | |
---|---|---|
SHA256 | 451e0db87f56a12776ef251ea2033ad9a7bff00442a66810a6c57556a214a0f9 |
|
MD5 | 7ccf0fe093ddc9aaf7b6097b5cc483e8 |
|
BLAKE2b-256 | c8c66ea2509557d24b86614d6beeda333b87de199627b60e3813fabe74f6f167 |