Nothing
## ----include = FALSE----------------------------------------------------------
knitr::opts_chunk$set(
collapse = TRUE,
comment = "#>"
)
## ----setup--------------------------------------------------------------------
library(geographiclib)
## ----locations----------------------------------------------------------------
# Major cities including Southern Hemisphere
cities <- cbind(
lon = c(151.21, 147.32, -0.13, -74.01, 139.69, -43.17, -68.30, 166.67),
lat = c(-33.87, -42.88, 51.51, 40.71, 35.69, -22.91, -54.80, -77.85)
)
rownames(cities) <- c("Sydney", "Hobart", "London", "New York",
"Tokyo", "Rio", "Ushuaia", "McMurdo")
# Antarctic stations
antarctic <- cbind(
lon = c(166.67, 77.85, -68.13, 39.58, 0),
lat = c(-77.85, -67.60, -67.57, -69.41, -90)
)
rownames(antarctic) <- c("McMurdo", "Davis", "Palmer", "Progress", "South Pole")
## ----inverse-basic------------------------------------------------------------
# Sydney to London
geodesic_inverse(c(151.21, -33.87), c(-0.13, 51.51))
## ----azimuth-examples---------------------------------------------------------
# Various directions from Sydney
destinations <- rbind(
"Due North" = c(151.21, -23), # Same longitude, further north
"Due East" = c(170, -33.87), # Same latitude, further east
"Due South" = c(151.21, -50), # Same longitude, further south
"Due West" = c(130, -33.87) # Same latitude, further west
)
sydney <- c(151.21, -33.87)
results <- geodesic_inverse(
cbind(rep(sydney[1], 4), rep(sydney[2], 4)),
destinations
)
data.frame(
destination = rownames(destinations),
azimuth = round(results$azi1, 1),
distance_km = round(results$s12 / 1000)
)
## ----inverse-cities-----------------------------------------------------------
# Distances from Sydney to other cities
sydney_idx <- 1
results <- geodesic_inverse(
cbind(rep(cities[sydney_idx, 1], nrow(cities) - 1),
rep(cities[sydney_idx, 2], nrow(cities) - 1)),
cities[-sydney_idx, ]
)
data.frame(
from = "Sydney",
to = rownames(cities)[-sydney_idx],
distance_km = round(results$s12 / 1000),
bearing = round(results$azi1, 1)
)
## ----inverse-antarctic--------------------------------------------------------
# Distances from South Pole to Antarctic stations
pole <- c(0, -90)
results <- geodesic_inverse(
cbind(rep(pole[1], nrow(antarctic) - 1),
rep(pole[2], nrow(antarctic) - 1)),
antarctic[-5, ] # Exclude South Pole itself
)
data.frame(
station = rownames(antarctic)[-5],
distance_from_pole_km = round(results$s12 / 1000),
bearing = round(results$azi1, 1)
)
## ----direct-basic-------------------------------------------------------------
# Travel 1000 km due north from Sydney
geodesic_direct(c(151.21, -33.87), azi = 0, s = 1000000)
## ----direct-circle------------------------------------------------------------
# Points 1000 km from Sydney in all directions
bearings <- seq(0, 330, by = 30)
results <- geodesic_direct(c(151.21, -33.87), azi = bearings, s = 1000000)
data.frame(
bearing = bearings,
direction = c("N", "NNE", "ENE", "E", "ESE", "SSE",
"S", "SSW", "WSW", "W", "WNW", "NNW"),
lon = round(results$lon2, 2),
lat = round(results$lat2, 2)
)
## ----direct-antarctic---------------------------------------------------------
# Plan traverse from McMurdo heading inland
mcmurdo <- c(166.67, -77.85)
# Head south-southwest at 195° bearing
waypoints <- geodesic_direct(
mcmurdo,
azi = 195,
s = c(0, 100000, 200000, 300000, 400000, 500000)
)
data.frame(
distance_km = waypoints$s12 / 1000,
lon = round(waypoints$lon2, 2),
lat = round(waypoints$lat2, 2)
)
## ----path-basic---------------------------------------------------------------
path <- geodesic_path(c(151.21, -33.87), c(-0.13, 51.51), n = 12)
path
# The path crosses into the Northern Hemisphere via Asia
## ----path-antarctic-----------------------------------------------------------
# Path around Antarctica at 70°S
start <- c(0, -70)
end <- c(180, -70) # Go halfway around
# This will follow a geodesic, not a parallel of latitude!
path <- geodesic_path(start, end, n = 10)
path
# Note: lat varies slightly - geodesics don't follow parallels
## ----path-transantarctic------------------------------------------------------
# McMurdo to Palmer Station (across the continent)
path <- geodesic_path(c(166.67, -77.85), c(-68.13, -67.57), n = 10)
path
## ----matrix-cities------------------------------------------------------------
# Distance matrix for all cities (km)
dist_matrix <- geodesic_distance_matrix(cities) / 1000
round(dist_matrix)
## ----matrix-antarctic---------------------------------------------------------
# Distances between Antarctic stations
dist_matrix <- geodesic_distance_matrix(antarctic) / 1000
round(dist_matrix)
## ----matrix-cross-------------------------------------------------------------
# Distance from each city to each Antarctic station
dist_matrix <- geodesic_distance_matrix(cities, antarctic) / 1000
rownames(dist_matrix) <- rownames(cities)
colnames(dist_matrix) <- rownames(antarctic)
round(dist_matrix)
## ----rhumb-compare------------------------------------------------------------
# Sydney to London: geodesic vs rhumb
start <- c(151.21, -33.87)
end <- c(-0.13, 51.51)
geo_result <- geodesic_inverse(start, end)
rhumb_result <- rhumb_inverse(start, end)
data.frame(
method = c("Geodesic", "Rhumb"),
distance_km = round(c(geo_result$s12, rhumb_result$s12) / 1000),
starting_bearing = round(c(geo_result$azi1, rhumb_result$azi12), 1),
extra_distance_km = c(0, round((rhumb_result$s12 - geo_result$s12) / 1000))
)
## ----rhumb-eastwest-----------------------------------------------------------
# Due east from Sydney (along parallel)
start <- c(151.21, -33.87)
end <- c(170, -33.87)
geo <- geodesic_inverse(start, end)
rhumb <- rhumb_inverse(start, end)
data.frame(
method = c("Geodesic", "Rhumb"),
distance_km = round(c(geo$s12, rhumb$s12) / 1000, 3),
bearing = round(c(geo$azi1, rhumb$azi12), 3)
)
# Nearly identical for E-W travel
## ----rhumb-path---------------------------------------------------------------
# Rhumb path vs geodesic path: Sydney to Cape Town
start <- c(151.21, -33.87)
end <- c(18.42, -33.92)
geo_path <- geodesic_path(start, end, n = 6)
rhumb_path_result <- rhumb_path(start, end, n = 6)
data.frame(
type = rep(c("Geodesic", "Rhumb"), each = 6),
point = rep(1:6, 2),
lon = c(geo_path$lon, rhumb_path_result$lon),
lat = round(c(geo_path$lat, rhumb_path_result$lat), 2)
)
# Note: rhumb line maintains more constant latitude
## ----polygon-area-------------------------------------------------------------
# Approximate Ross Ice Shelf boundary
ross_ice_shelf <- cbind(
lon = c(158, 170, -175, -160, -150, -158, -170, 170, 158),
lat = c(-77, -78, -78.5, -79, -78, -77.5, -77, -77, -77)
)
result <- polygon_area(ross_ice_shelf)
result
# Area in km²
abs(result$area) / 1e6
## ----polygon-multi------------------------------------------------------------
# Compare areas of different regions
pts <- cbind(
lon = c(
# Tasmania (approximate)
144, 148, 148, 144, 144,
# South Island NZ (approximate)
166, 174, 174, 166, 166
),
lat = c(
-40, -40, -44, -44, -40,
-41, -41, -47, -47, -41
)
)
result <- polygon_area(pts, id = c(rep(1, 5), rep(2, 5)))
data.frame(
region = c("Tasmania", "South Island NZ"),
area_km2 = round(abs(result$area) / 1e6)
)
## ----polygon-winding----------------------------------------------------------
# Counter-clockwise vs clockwise
ccw <- cbind(lon = c(0, 1, 1, 0), lat = c(0, 0, 1, 1))
cw <- cbind(lon = c(0, 0, 1, 1), lat = c(0, 1, 1, 0))
data.frame(
winding = c("Counter-clockwise", "Clockwise"),
area_km2 = c(polygon_area(ccw)$area, polygon_area(cw)$area) / 1e6
)
# Areas have opposite signs
## ----exact-vs-fast------------------------------------------------------------
# Compare accuracy and speed
start <- c(151.21, -33.87)
end <- c(-0.13, 51.51)
exact <- geodesic_inverse(start, end)
fast <- geodesic_inverse_fast(start, end)
data.frame(
version = c("Exact", "Fast"),
distance_m = c(exact$s12, fast$s12),
difference_mm = c(0, (fast$s12 - exact$s12) * 1000)
)
# Difference is typically nanometers - negligible for most uses
## ----performance, eval=FALSE--------------------------------------------------
# # Generate 10,000 random point pairs
# n <- 10000
# pts1 <- cbind(runif(n, -180, 180), runif(n, -90, 90))
# pts2 <- cbind(runif(n, -180, 180), runif(n, -90, 90))
#
# system.time(geodesic_inverse(pts1, pts2))
# #> user system elapsed
# #> 0.05 0.00 0.05
#
# system.time(geodesic_inverse_fast(pts1, pts2))
# #> user system elapsed
# #> 0.04 0.00 0.04
## ----intersect-closest--------------------------------------------------------
# Two geodesics starting from different points
# Geodesic X: from (0, 0) heading NE at 45°
# Geodesic Y: from (1, 0) heading NW at 315°
result <- geodesic_intersect(c(0, 0), 45, c(1, 0), 315)
result
# The intersection is found ahead on both geodesics
data.frame(
geodesic = c("X", "Y"),
start = c("(0, 0)", "(1, 0)"),
azimuth = c(45, 315),
displacement_km = round(c(result$x, result$y) / 1000, 2)
)
## ----intersect-segment--------------------------------------------------------
# Two crossing segments
# Segment X: from (0, -0.5) to (0, 0.5) - roughly N-S
# Segment Y: from (-0.5, 0) to (0.5, 0) - roughly E-W
result <- geodesic_intersect_segment(
c(0, -0.5), c(0, 0.5), # Segment X endpoints
c(-0.5, 0), c(0.5, 0) # Segment Y endpoints
)
result
# segmode = 0 means intersection is within both segments
if (result$segmode == 0) {
cat("Segments intersect at:", result$lat, result$lon, "\n")
} else {
cat("Segments do not intersect (extended intersection at:",
result$lat, result$lon, ")\n")
}
## ----intersect-segment-parallel-----------------------------------------------
# Two parallel segments that don't intersect
result <- geodesic_intersect_segment(
c(0, 0), c(0, 1), # Segment X: lon=0, lat 0 to 1
c(1, 0), c(1, 1) # Segment Y: lon=1, lat 0 to 1
)
cat("segmode:", result$segmode, "- segments do not intersect\n")
cat("Closest point of intersection would be at:", result$lat, result$lon, "\n")
## ----intersect-next-----------------------------------------------------------
# Two geodesics crossing at the origin
# Find where they cross again
next_int <- geodesic_intersect_next(c(0, 0), 45, 90)
data.frame(
intersection = c("origin", "next"),
lat = c(0, next_int$lat),
lon = c(0, next_int$lon),
distance_km = c(0, round(abs(next_int$x) / 1000))
)
## ----intersect-all------------------------------------------------------------
# Find all intersections within 5,000 km
all_ints <- geodesic_intersect_all(
c(0, 0), 30, # Geodesic X: from origin, azimuth 30°
c(10, 0), 330, # Geodesic Y: from (10, 0), azimuth 330°
maxdist = 5e6 # Search radius: 5,000 km
)
cat("Found", nrow(all_ints), "intersections within 5,000 km\n")
all_ints
## ----intersect-routes---------------------------------------------------------
# Route 1: Sydney to London (via typical great circle)
sydney <- c(151.21, -33.87)
london <- c(-0.13, 51.51)
route1 <- geodesic_inverse(sydney, london)
# Route 2: Tokyo to São Paulo
tokyo <- c(139.69, 35.69)
sao_paulo <- c(-46.63, -23.55)
route2 <- geodesic_inverse(tokyo, sao_paulo)
# Find intersection of extended routes
crossing <- geodesic_intersect(
sydney, route1$azi1,
tokyo, route2$azi1
)
cat("Routes cross at:", round(crossing$lat, 2), "lat,",
round(crossing$lon, 2), "lon\n")
# Distance from each origin to crossing
cat("From Sydney:", round(crossing$x / 1000), "km\n")
cat("From Tokyo:", round(crossing$y / 1000), "km\n")
## ----intersect-vectorized-----------------------------------------------------
# Multiple intersection problems at once
results <- geodesic_intersect(
cbind(c(0, 0, 0), c(0, 10, 20)), # Three starting points for X
c(45, 60, 75), # Three azimuths for X
cbind(c(5, 5, 5), c(0, 10, 20)), # Three starting points for Y
c(315, 300, 285) # Three azimuths for Y
)
data.frame(
problem = 1:3,
lat = round(results$lat, 4),
lon = round(results$lon, 4),
dist_x_km = round(results$x / 1000, 1),
dist_y_km = round(results$y / 1000, 1)
)
## ----nn-basic-----------------------------------------------------------------
# Australian cities dataset
oz_cities <- cbind(
lon = c(151.21, 144.96, 153.03, 115.86, 138.60, 149.13),
lat = c(-33.87, -37.81, -27.47, -31.95, -34.93, -35.28)
)
rownames(oz_cities) <- c("Sydney", "Melbourne", "Brisbane",
"Perth", "Adelaide", "Canberra")
# Find 3 nearest cities to Canberra
query <- c(149.13, -35.28) # Canberra
result <- geodesic_nn(oz_cities, query, k = 3)
# Show nearest cities
data.frame(
rank = 1:3,
city = rownames(oz_cities)[result$index[, 1]],
distance_km = round(result$distance[, 1] / 1000, 1)
)
## ----nn-multiple--------------------------------------------------------------
# New cities to find neighbors for
queries <- cbind(
lon = c(147.32, 130.84),
lat = c(-42.88, -12.46)
)
rownames(queries) <- c("Hobart", "Darwin")
result <- geodesic_nn(oz_cities, queries, k = 2)
# Results for each query
for (i in 1:nrow(queries)) {
cat(rownames(queries)[i], "- nearest cities:\n")
for (j in 1:2) {
cat(" ", j, ". ", rownames(oz_cities)[result$index[j, i]],
" (", round(result$distance[j, i] / 1000), " km)\n", sep = "")
}
}
## ----nn-radius----------------------------------------------------------------
# Find all cities within 500 km of Adelaide
adelaide <- c(138.60, -34.93)
within_500km <- geodesic_nn_radius(oz_cities, adelaide, radius = 500000)
if (nrow(within_500km[[1]]) > 0) {
data.frame(
city = rownames(oz_cities)[within_500km[[1]]$index],
distance_km = round(within_500km[[1]]$distance / 1000, 1)
)
}
## ----nn-self------------------------------------------------------------------
# Find each city's nearest neighbor (excluding itself)
result <- geodesic_nn(oz_cities, oz_cities, k = 2)
# The second neighbor (k=2) is the nearest *other* city
data.frame(
city = rownames(oz_cities),
nearest = rownames(oz_cities)[result$index[2, ]],
distance_km = round(result$distance[2, ] / 1000, 1)
)
## ----nn-distmat---------------------------------------------------------------
# Distance from each Australian city to key world cities
world_cities <- cities[c("Sydney", "London", "New York", "Tokyo"), ]
# Get distances (k = all world cities)
result <- geodesic_nn(world_cities, oz_cities, k = nrow(world_cities))
# Format as distance matrix (km)
dist_mat <- matrix(
round(result$distance / 1000),
nrow = nrow(world_cities),
dimnames = list(rownames(world_cities), rownames(oz_cities))
)
dist_mat
## ----nn-performance, eval=FALSE-----------------------------------------------
# # Example with larger dataset (not run)
# set.seed(42)
# large_dataset <- cbind(
# lon = runif(10000, 110, 155),
# lat = runif(10000, -45, -10)
# )
# queries <- cbind(
# lon = runif(100, 110, 155),
# lat = runif(100, -45, -10)
# )
#
# system.time(geodesic_nn(large_dataset, queries, k = 10))
# #> user system elapsed
# #> 0.15 0.00 0.15
Any scripts or data that you put into this service are public.
Add the following code to your website.
For more information on customizing the embed code, read Embedding Snippets.