Получение расстояния вождения между двумя точками (широта, долгота) с помощью R и Google Map API
Я пытаюсь получить расстояние между двумя точками с учетом широты / долготы. Я могу вручную поместить их в карту Google и получить проезжающее расстояние, но я хочу сделать все это программно.
Я предполагаю, что JavaScript - это язык, на котором можно пойти. Но я не знаю JavaScript, и я довольно знаком с использованием R. Я бы предпочел сделать это в R, так как я делаю весь анализ данных в R.
Я ищу расстояние по дороге, а не расстояние до мух. После нескольких часов попыток я написал следующую функцию в R ( это помогло и это). У вас есть какой-нибудь лучший способ получить расстояние в рамках этой функции или что-то очень простое?
library(XML)
latlon2ft <- function(origin,destination)
{
xml.url <- paste0('http://maps.googleapis.com/maps/api/distancematrix/xml?origins=',origin,'&destinations=',destination,'&mode=driving&sensor=false')
xmlfile <- xmlTreeParse(xml.url)
xmltop = xmlRoot(xmlfile)
distance <- xmltop[['row']][[1]][5][1][['distance']][['value']][[1]]
distance <- as.numeric(unclass(distance)[['value']])
ft <- distance*3.28084 # FROM METER TO FEET
return(ft)
}
latlon2ft(origin='37.193489,-121.07395',destination='37.151616,-121.046586')
РЕЗУЛЬТАТ = 17224,41
5 ответов
Я автор gmapsdistance
пакет, чтобы сделать именно это. Это доступно на CRAN. Вы можете использовать функцию следующим образом:
results = gmapsdistance(origin = "38.1621328+24.0029257",
destination = "37.9908372+23.7383394",
mode = "walking") results
# $Time
# [1] 30025
#
# $Distance
# [1] 39507
#
# $Status
# [1] "OK"
Вы также можете включить векторы происхождения и назначения и получить полученную матрицу расстояний. Он также поддерживает направления и имеет несколько вариантов:
results = gmapsdistance(origin = c("Washington+DC", "New+York+NY", "Seattle+WA", "Miami+FL"),
destination = c("Los+Angeles+CA", "Austin+TX", "Chicago+IL", "Philadelphia+PA"),
mode = "bicycling",
departure = 1514742000)
results
# $Time
# or Time.Los+Angeles+CA Time.Austin+TX Time.Chicago+IL Time.Philadelphia+PA
# 1 Washington+DC 856621 535146 247765 54430
# 2 New+York+NY 917486 596011 308630 32215
# 3 Seattle+WA 374692 678959 674989 956702
# 4 Miami+FL 829039 416667 452035 411283
#
# $Distance
# or Distance.Los+Angeles+CA Distance.Austin+TX Distance.Chicago+IL Distance.Philadelphia+PA
# 1 Washington+DC 4567470 2838519 1303067 266508
# 2 New+York+NY 4855086 3126136 1590684 160917
# 3 Seattle+WA 1982354 3562970 3588297 5051951
# 4 Miami+FL 4559205 2279966 2381610 2169382
#
# $Status
# or status.Los+Angeles+CA status.Austin+TX status.Chicago+IL status.Philadelphia+PA
# 1 Washington+DC OK OK OK OK
# 2 New+York+NY OK OK OK OK
# 3 Seattle+WA OK OK OK OK
# 4 Miami+FL OK OK OK OK
Тебе нужно RCurl
или эквивалент здесь.
library(XML)
library(bitops)
library(RCurl)
latlon2ft <- function(origin,destination){
xml.url <- paste0('http://maps.googleapis.com/maps/api/distancematrix/xml?origins=',origin,'&destinations=',destination,'&mode=driving&sensor=false')
xmlfile <- xmlParse(getURL(xml.url))
dist <- xmlValue(xmlChildren(xpathApply(xmlfile,"//distance")[[1]])$value)
distance <- as.numeric(sub(" km","",dist))
ft <- distance*3.28084 # FROM METER TO FEET
return(ft)
}
latlon2ft(origin='37.193489,-121.07395',destination='37.151616,-121.046586')
Результат:
[1] 17224.41
Мне нужно было рассчитать расстояние в пути для нескольких адресов, поэтому я написал для него небольшую функцию и поместил ее в небольшой пакет. Вы можете найти его в моем репозитории GitHub: https://github.com/JanMultmeier/GeoData/blob/master/GeoDataPackage/R/GetDist.R
Это должно заставить это бежать:
require(devtools)
install_github("JanMultmeier/GeoData/GeoDataPackage")
library(GeoData)
getDist(from="1 Infinity Loop, Cupertino, CA 95014", to="1600 Amphitheatre Pkwy, Mountain View, CA 94043",modus="driving",get="distance")
Должен вернуться 14,8 км.
Barryhunter уже намекнул на ограничение использования Google, которое связывает использование этого API с отображением результатов на карте Google.
Надеюсь, что это по-прежнему помогает некоторым людям, которые наткнулись на этот пост (как я)...
На момент написания статьи Renjin (интерпретатор R на основе Java) не располагал множеством пакетов, помогающих решить эту проблему. Вот реализация, которая не зависит от дополнительных пакетов.
# Computes the distance between two locations in meters. This uses an online
# map API and therefore an Internet connection is required for an accurate
# result. If no connection is found, this will use the Haversine formula
# to provide a rough estimate for the distance.
#
# @param src The source latitude and longitude.
# @param dst The destination latitude and longitude.
# @param mode Driving, cycling, walking, etc.
distance <- function( lat1, lon1, lat2, lon2, mode = 'driving' ) {
lat1 = as.numeric( lat1 )
lon1 = as.numeric( lon1 )
lat2 = as.numeric( lat2 )
lon2 = as.numeric( lon2 )
# Create the URL to use to get the distance data.
url = paste0(
'https://maps.googleapis.com/maps/api/distancematrix/xml?',
'origins=', lat1,
',', lon1,
'&destinations=', lat2,
',', lon2,
'&mode=', mode,
'&sensor=false'
)
tryCatch({
# Download the XML document with distance information.
xml = readLines( url )
# The <value> element immediately follows the distance element.
value = xml[ grep( "<distance>", xml ) + 1 ]
# Obtain the distance in meters.
meters = sub( ".*>(.*?)<.*", "\\1", value )
# Return the distance.
as.numeric( meters )
},
warning = function( w ) {
haversine( lat1, lon1, lat2, lon2 )
},
error = function( e ) {
haversine( lat1, lon1, lat2, lon2 )
})
}
# Computes distance using Haversine formula.
#
# Returns the result in meters.
haversine <- function( lat1, lon1, lat2, lon2, radius = 6371 ) {
# Convert decimal degrees to radians
lon1 = lon1 * pi / 180
lon2 = lon2 * pi / 180
lat1 = lat1 * pi / 180
lat2 = lat2 * pi / 180
# Haversine formula
dlon = lon2 - lon1
dlat = lat2 - lat1
a = sin(dlat/2)**2 + cos(lat1) * cos(lat2) * sin(dlon/2)**2
c = 2 * atan2(sqrt(a), sqrt(1-a))
return( radius * c * 1000 )
}
Выход:
distance( '44.5646', '-123.2620', '41.2861', '-124.0902' )
[1] 495892
distance( 44.5646, -123.2620, 41.2861, -124.0902, mode='walking' )
[1] 487715
Перевод из метра в футы - упражнение для читателя.
Я написал googleway
пакет для этого с помощью Google Maps API
В частности, google_directions()
функция даст вам проезжающие расстояния, направления, маршруты, ноги, шаги и т. д. google_distance()
Функция даст вам матрицу расстояний для всех мест происхождения / назначения
Вам нужен ключ API Google, чтобы использовать их API
library(googleway)
## your valid API key
key <- "your_api_key_here"
directions <- google_directions(origin = c(37.193489,-121.07395),
destination = c(37.151616,-121.046586),
key = key,
simplify = T)
directions$routes$legs
# [[1]]
# distance.text distance.value duration.text duration.value duration_in_traffic.text duration_in_traffic.value end_address
# 1 5.2 km 5250 3 mins 161 3 mins 156 I-5, Gustine, CA 95322, USA
# end_location.lat end_location.lng start_address start_location.lat start_location.lng
# 1 37.15162 -121.0466 I-5, Gustine, CA 95322, USA 37.19349 -121.074
# steps
# 1 5.2 km, 5250, 3 mins, 161, 37.1516163, -121.0465852, Head <b>southeast</b> on <b>I-5 S</b>, ij_bFfg~aVpBgA`DkB~FeDbIwEpEgCtaAsj@nAs@lDqBxIaF~FgDlHcEjC{AdFuCrBkAhC{A|A{@|A}@bAk@rBkArBkA|A{@`DiB|A}@vDwBdAm@dAm@rBkA|A{@zA{@~J{FpC_B~A}@tBkAjHeEvGuDlMmHtBkAVO, 37.1934864, -121.0739565, DRIVING
# traffic_speed_entry via_waypoint
# 1 NULL NULL
google_distance(origins = list(c(37.193489,-121.07395)),
destinations = list(c(37.151616,-121.046586)),
key = key,
simplify = T,
units = "imperial")
# $destination_addresses
# [1] "I-5, Gustine, CA 95322, USA"
#
# $origin_addresses
# [1] "I-5, Gustine, CA 95322, USA"
#
# $rows
# elements
# 1 3.3 mi, 5250, 3 mins, 161, 3 mins, 157, OK
#
# $status
# [1] "OK"
Учитывая google_directions()
функция возвращает полилинию (линию, которую вы получаете на Google Maps при поиске маршрута), мы можем нанести ее на карту Google
key <- 'your_map_api_key'
df_route <- decode_pl(directions$routes$overview_polyline$points)
google_map(data = df_route, key = key, height = 800, search_box = T) %>%
add_markers()
## or you can use `add_polyline()` to view the entire line