MongoDB имеет встроенную поддержку для пространственных индексов , поэтому все, что вам нужно сделать, это загрузить точки в правильном формате, создать пространственный индекс и затем выполнить запросы.
Для быстрого примера я загрузил центральные точки для всех 50 состояний в оболочке mongo:
> db.places.ensureIndex({loc: "2d"})
> db.places.save({name: "AK", loc: {long: -152.2683, lat: 61.3850}})
> db.places.save({name: "AL", loc: {long: -86.8073, lat: 32.7990}})
> db.places.save({name: "AR", loc: {long: -92.3809, lat: 34.9513}})
> db.places.save({name: "AS", loc: {long: -170.7197, lat: 14.2417}})
> ...
Далее, чтобы запросить 6 ближайших точек к указанному местоположению :
> db.places.find({loc: { $near: {long: -90, lat: 50}}}).limit(6)
{"name" : "WI", "loc" : { "long" : -89.6385, "lat" : 44.2563 } }
{"name" : "MN", "loc" : { "long" : -93.9196, "lat" : 45.7326 } }
{"name" : "MI", "loc" : { "long" : -84.5603, "lat" : 43.3504 } }
{"name" : "IA", "loc" : { "long" : -93.214, "lat" : 42.0046 } }
{"name" : "IL", "loc" : { "long" : -89.0022, "lat" : 40.3363 } }
{"name" : "ND", "loc" : { "long" : -99.793, "lat" : 47.5362 } }
Далее, чтобы запросить все точки в пределах 10 км от заданного местоположения .Поскольку я вычисляю ближайшие состояния, я буду использовать 888 км (что примерно равно 8 градусам широты):
> db.places.find({loc: { $near: {long: -90, lat: 50}, $maxDistance: 8}})
{"name" : "WI", "loc" : { "long" : -89.6385, "lat" : 44.2563 } }
{"name" : "MN", "loc" : { "long" : -93.9196, "lat" : 45.7326 } }
Поскольку один градус широты равен приблизительно 111.12км , вы бы использовали $maxDistance: 0.08999
для представления 10 км для вашего приложения.
Обновлено По умолчанию MongoDB предполагает "идеализированную модель плоской земли", но это приводит к неточностям, посколькулинии долготы сходятся на полюсах. MongoDB версии 1.7+ поддерживает вычисления сферического расстояния , что обеспечивает повышенную точность.
Ниже приведен пример выполнения вышеуказанного запроса с использованием сферического расстояния.maxDistance
в радианах, поэтому нам нужно разделить на средний радиус Земли:
> db.runCommand({geoNear: "places", near: [-90, 50], spherical: true,
maxDistance: 800/6378});
(summarizing results as they're too verbose to include)
"MN" dis: 0.087..
"WI" dis: 0.100..
"ND" dis: 0.120..