如何在 PHP 中高效查找 JSON 地理数据中距离最近的站点 ID

1次阅读

如何在 PHP 中高效查找 JSON 地理数据中距离最近的站点 ID

本文详解如何解析 admiralty tide api 返回的 geojson 数据,基于给定经纬度计算球面距离,精准定位最近站点并提取其 `id` 字段,适用于潮汐查询、地理服务等实际场景。

在处理地理坐标数据(如 Admiralty 的潮汐站列表)时,简单使用欧氏距离(Pythagorean Theorem)会因地球曲率导致显著误差——尤其当坐标跨纬度较大时。正确做法是采用大圆距离公式(Haversine 或球面余弦定理),它基于地球近似球体模型,单位精度可达海里(nmi)、公里(km)或英里(mi)。

以下是一个完整、健壮、可直接部署的 php 教程实现:

✅ 正确解析 GeojsON 结构

Admiralty API 返回的是标准 Geojson FeatureCollection,其核心结构为:

{   "type": "FeatureCollection",   "features": [     {       "type": "Feature",       "geometry": { "type": "Point", "coordinates": [-4.702540, 51.672791] },       "properties": { "Id": "1603", "Name": "TENBY", ... }     }   ] }

注意:coordinates 数组顺序为 [longitude, latitude](先经度后纬度),这与常见 lat/lng 变量命名相反,极易出错——务必确认 $lng = $feature->geometry->coordinates[0],$lat = $feature->geometry->coordinates[1]。

立即学习PHP免费学习笔记(深入)”;

✅ 使用球面余弦距离函数(高精度 & 轻量)

以下函数基于球面余弦定理,兼顾精度与性能,支持 nmi(默认)、km、mi 单位:

function getDistance($from_lat, $from_lng, $to_lat, $to_lng, $unit = 'nmi', $decimals = 2) {     // 将角度转为弧度进行三角运算     $lat1 = deg2rad($from_lat);     $lng1 = deg2rad($from_lng);     $lat2 = deg2rad($to_lat);     $lng2 = deg2rad($to_lng);      // 球面余弦距离公式(单位:度)     $degrees = rad2deg(acos(         sin($lat1) * sin($lat2) +         cos($lat1) * cos($lat2) * cos($lng2 - $lng1)     ));      // 按单位换算为实际距离(地球平均半径换算系数)     $conversion = match($unit) {         'km'  => 111.13384,  // 1° ≈ 111.13 km         'mi'  => 69.05482,   // 1° ≈ 69.05 mi         'nmi' => 59.97662,   // 1° ≈ 59.98 nmi (航海常用)         default => 59.97662     };      return round($degrees * $conversion, $decimals); }

? 提示:PHP 8.0+ 推荐使用 match 表达式;若用旧版 PHP,请替换为 switch

✅ 主逻辑:单次遍历 + 实时比较(O(n) 时间复杂度)

避免冗余循环和中间数组(如原代码中多层 foreach 嵌套),直接遍历 features,动态维护最小距离及对应站点:

// 获取实时站点数据(生产环境请添加错误处理与缓存) $json = file_get_contents('https://easytide.admiralty.co.uk/Home/GetStations'); if ($json === false) {     die("❌ Failed to fetch stations from Admiralty API."); } $stations = json_decode($json);  // 设置参考位置(例如:朴茨茅斯城堡,用于测试) $user_location = ['lat' => 50.778423, 'lng' => -1.087805];  $closest = null; $min_distance = null;  foreach ($stations->features as $index => $feature) {     // ✅ 关键:GeoJSON coordinates = [lng, lat]     $lng = $feature->geometry->coordinates[0];     $lat = $feature->geometry->coordinates[1];      $distance = getDistance(         $user_location['lat'], $user_location['lng'],         $lat, $lng,         'nmi', 4     );      if ($min_distance === null || $distance < $min_distance) {         $min_distance = $distance;         $closest = [             'id'       => $feature->properties->Id,             'name'     => $feature->properties->Name,             'distance' => $distance,             'unit'     => 'nmi',             'index'    => $index         ];     } }  // ✅ 输出结果(可用于后续 API 请求) if ($closest) {     echo "✅ Closest station: {$closest['name']} (ID: {$closest['id']})n";     echo "? Distance: {$closest['distance']} {$closest['unit']}n";     echo "? Use ID '{$closest['id']}' for tide data: n";     echo "   file_get_contents('https://easytide.admiralty.co.uk/Home/GetTideTimes?stationId={$closest['id']}')"; } else {     echo "⚠️  No stations found."; }

⚠️ 注意事项与最佳实践

  • HTTPS 与错误处理:file_get_contents() 在无响应或超时时会返回 false,务必检查并加入重试或降级逻辑;
  • CORS / API 限制:Admiralty 站点接口可能限制请求频率或来源,生产环境建议服务端代理或加缓存(如 redis 存储 GetStations 结果,TTL 1 小时);
  • 精度取舍:对百公里内应用,Haversine 公式已足够;若需亚米级精度(如测绘),应使用 Vincenty 或 GeographicLib;
  • 性能优化:若站点数超万级,可预构建空间索引(如 GeoHash 或 mysql POINT + ST_Distance_Sphere),但 Admiralty 当前约 300+ 站点,线性扫描完全足够;
  • 安全提醒:勿将 file_get_contents() 直接用于用户可控 URL,防止 SSRF;此处 URL 固定,风险可控。

✅ 总结

你真正需要的不是“嵌套循环遍历子数组”,而是:

  1. 理解 GeoJSON 坐标顺序([lng, lat]);
  2. 选用正确的地理距离算法(拒绝平面直角距离);
  3. 一次遍历 + 状态保持(无需存储所有距离,节省内存);
  4. 结构化返回结果(含 Id、Name、distance,便于下游调用)。

现在,你已掌握从 Admiralty API 动态获取最近潮汐站 ID 的完整链路——下一步,即可用该 Id 请求详细潮时数据,构建完整的海洋信息服务。

text=ZqhQzanResources