0

0

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

花韻仙語

花韻仙語

发布时间:2026-02-07 16:49:39

|

244人浏览过

|

来源于php中文网

原创

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

本文详解如何解析 admiralty tide api 的 geojson 数据,结合 haversine 公式精确计算球面距离,并快速定位最近站点的 `id` 字段,适用于潮位查询、地理服务集成等实际场景。

在处理地理坐标数据(如潮位站、气象站或地图标记)时,一个常见需求是:给定用户当前经纬度,从远程 API 返回的大量站点中找出物理距离最近的一个,并提取其唯一标识(如 properties->Id)。原始代码存在多重嵌套循环、错误的数组遍历路径(误将整个 $locations 当作多维数组逐层 foreach),且使用欧氏距离(平面直角坐标系)计算经纬度——这在地球曲面上会产生显著误差(尤其跨纬度时)。下面提供一套健壮、可复用、符合地理精度要求的解决方案。

✅ 正确解析结构 & 遍历逻辑

Admiralty 的 /Home/GetStations 接口返回标准 GeoJSON FeatureCollection,其核心结构为:

{
  "type": "FeatureCollection",
  "features": [
    {
      "type": "Feature",
      "geometry": { "type": "Point", "coordinates": [lng, lat] },
      "properties": { "Id": "0065", "Name": "PORTSMOUTH", ... }
    }
  ]
}

关键点:

  • coordinates 是 [longitude, latitude](注意顺序!X 轴为经度,Y 轴为纬度);
  • 应直接遍历 $stations->features(对象模式)或 $stations['features'](关联数组模式),无需深层嵌套 foreach
  • 使用 json_decode($json, false)(默认 false)返回对象,更符合 GeoJSON 规范访问习惯。

✅ 使用 Haversine 公式计算真实球面距离

地球是球体,两点间最短路径是大圆弧。以下函数封装了高精度 Haversine 计算,支持公里(km)、英里(mi)和海里(nmi)三种单位:

法语写作助手
法语写作助手

法语助手旗下的AI智能写作平台,支持语法、拼写自动纠错,一键改写、润色你的法语作文。

下载

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

function getDistance(float $from_lat, float $from_lng, float $to_lat, float $to_lng, string $unit = 'nmi', int $decimals = 2): float {
    $lat1 = deg2rad($from_lat);
    $lng1 = deg2rad($from_lng);
    $lat2 = deg2rad($to_lat);
    $lng2 = deg2rad($to_lng);

    $dlat = $lat2 - $lat1;
    $dlng = $lng2 - $lng1;

    $a = sin($dlat / 2) * sin($dlat / 2) +
         cos($lat1) * cos($lat2) *
         sin($dlng / 2) * sin($dlng / 2);
    $c = 2 * atan2(sqrt($a), sqrt(1 - $a));

    // 地球平均半径(km)
    $earth_radius_km = 6371.0;
    $distance_km = $earth_radius_km * $c;

    switch (strtolower($unit)) {
        case 'km':  $distance = $distance_km; break;
        case 'mi':  $distance = $distance_km * 0.621371; break;
        case 'nmi': $distance = $distance_km * 0.539957; break;
        default:    $distance = $distance_km;
    }

    return round($distance, $decimals);
}
⚠️ 注意:原始答案中使用的 rad2deg(acos(...)) 方法在极近距离或数值精度边界下可能因浮点误差导致 acos() 参数略超 [-1,1] 范围而报错;Haversine 的 atan2 形式更鲁棒。

✅ 完整可运行示例(含错误处理)

 50.77842324616663,
    'lng' => -1.087804949548603
];

// 3. 遍历 features,计算距离,追踪最小值
$closest = null;
$min_distance = null;

foreach ($stations->features as $index => $feature) {
    // 安全检查:确保 geometry 和 coordinates 存在
    if (!isset($feature->geometry->coordinates[0], $feature->geometry->coordinates[1])) {
        continue;
    }

    $station_lng = (float)$feature->geometry->coordinates[0];
    $station_lat = (float)$feature->geometry->coordinates[1];

    $distance = getDistance(
        $user_location['lat'],
        $user_location['lng'],
        $station_lat,
        $station_lng,
        'nmi', // 单位:海里(航海常用)
        4
    );

    if ($min_distance === null || $distance < $min_distance) {
        $min_distance = $distance;
        $closest = [
            'distance'   => $distance,
            'index'      => $index,
            'id'         => $feature->properties->Id ?? 'N/A',
            'name'       => $feature->properties->Name ?? 'Unknown',
            'country'    => $feature->properties->Country ?? 'N/A'
        ];
    }
}

// 4. 输出结果(供后续调用,如 file_get_contents 构造新请求)
if ($closest) {
    echo "

✅ 最近站点信息

"; echo "
    "; echo "
  • ID: {$closest['id']}
  • "; echo "
  • 名称: {$closest['name']}
  • "; echo "
  • 距离: {$closest['distance']} 海里
  • "; echo "
  • 国家: {$closest['country']}
  • "; echo "
"; // ✅ 关键:这就是你后续请求所需的 ID echo "

下一步可使用 ID '{$closest['id']}' 请求潮位详情:

"; echo "
file_get_contents('https://easytide.admiralty.co.uk/Home/GetTideData?stationId={$closest['id']}')
"; } else { echo "⚠️ 未找到有效站点"; }

? 总结与最佳实践

  • 避免平面距离陷阱:永远不要对经纬度直接使用 (lat1-lat2)² + (lng1-lng2)²,它仅在赤道附近小范围近似有效;
  • 结构化遍历:GeoJSON 层级清晰,直击 $data->features 即可,无需多层 foreach 嵌套;
  • 防御性编程:始终检查 isset() 和 json_last_error(),API 数据格式可能变更;
  • 性能提示:若站点数达万级,建议预存为数据库并建立空间索引(如 MySQL POINT + ST_Distance_Sphere);
  • 单位一致性:getDistance() 函数明确区分输入(十进制度)与输出(指定单位),避免混淆。

此方案已验证可稳定对接 Admiralty Tide API,输出精准、结构清晰、易于扩展,是地理距离检索类任务的可靠基础模板。

相关文章

PHP速学教程(入门到精通)
PHP速学教程(入门到精通)

PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!

下载

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
mysql修改数据表名
mysql修改数据表名

MySQL修改数据表:1、首先查看数据库中所有的表,代码为:‘SHOW TABLES;’;2、修改表名,代码为:‘ALTER TABLE 旧表名 RENAME [TO] 新表名;’。php中文网还提供MySQL的相关下载、相关课程等内容,供大家免费下载使用。

670

2023.06.20

MySQL创建存储过程
MySQL创建存储过程

存储程序可以分为存储过程和函数,MySQL中创建存储过程和函数使用的语句分别为CREATE PROCEDURE和CREATE FUNCTION。使用CALL语句调用存储过程智能用输出变量返回值。函数可以从语句外调用(通过引用函数名),也能返回标量值。存储过程也可以调用其他存储过程。php中文网还提供MySQL创建存储过程的相关下载、相关课程等内容,供大家免费下载使用。

329

2023.06.21

mongodb和mysql的区别
mongodb和mysql的区别

mongodb和mysql的区别:1、数据模型;2、查询语言;3、扩展性和性能;4、可靠性。本专题为大家提供mongodb和mysql的区别的相关的文章、下载、课程内容,供大家免费下载体验。

282

2023.07.18

mysql密码忘了怎么查看
mysql密码忘了怎么查看

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql密码忘了怎么办呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

517

2023.07.19

mysql创建数据库
mysql创建数据库

MySQL是一个关系型数据库管理系统,由瑞典MySQL AB 公司开发,属于 Oracle 旗下产品。MySQL 是最流行的关系型数据库管理系统之一,在 WEB 应用方面,MySQL是最好的 RDBMS 应用软件之一。那么mysql怎么创建数据库呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

259

2023.07.25

mysql默认事务隔离级别
mysql默认事务隔离级别

MySQL是一种广泛使用的关系型数据库管理系统,它支持事务处理。事务是一组数据库操作,它们作为一个逻辑单元被一起执行。为了保证事务的一致性和隔离性,MySQL提供了不同的事务隔离级别。php中文网给大家带来了相关的教程以及文章欢迎大家前来学习阅读。

388

2023.08.08

sqlserver和mysql区别
sqlserver和mysql区别

SQL Server和MySQL是两种广泛使用的关系型数据库管理系统。它们具有相似的功能和用途,但在某些方面存在一些显著的区别。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

536

2023.08.11

mysql忘记密码
mysql忘记密码

MySQL是一种关系型数据库管理系统,关系数据库将数据保存在不同的表中,而不是将所有数据放在一个大仓库内,这样就增加了速度并提高了灵活性。那么忘记mysql密码我们该怎么解决呢?php中文网给大家带来了相关的教程以及其他关于mysql的文章,欢迎大家前来学习阅读。

631

2023.08.14

Golang处理数据库错误教程合集
Golang处理数据库错误教程合集

本专题整合了Golang数据库错误处理方法、技巧、管理策略相关内容,阅读专题下面的文章了解更多详细内容。

2

2026.02.06

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MySQL 教程
MySQL 教程

共48课时 | 2.1万人学习

MySQL 初学入门(mosh老师)
MySQL 初学入门(mosh老师)

共3课时 | 0.3万人学习

简单聊聊mysql8与网络通信
简单聊聊mysql8与网络通信

共1课时 | 823人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号