
PHP里用sqrt()和pow()算平面上两点距离
直接用勾股定理就行,不是非得调地理坐标函数。平面直角坐标系下,两点 (x1, y1) 和 (x2, y2) 的距离公式是 sqrt(pow($x2 - $x1, 2) + pow($y2 - $y1, 2))。
常见错误是漏掉括号导致运算优先级出错,比如写成 sqrt(pow($x2 - $x1, 2) + pow($y2 - $y1, 2)(少一个右括号),或者误用 abs() 替代平方——那算出来是曼哈顿距离,不是欧氏距离。
- 推荐写法:
$distance = sqrt(pow($x2 - $x1, 2) + pow($y2 - $y1, 2)); - 想省一次函数调用?可以用
**运算符(PHP 5.6+):sqrt(($x2 - $x1) ** 2 + ($y2 - $y1) ** 2) - 如果输入可能为字符串(比如表单提交),先用
(float)强转,否则pow('5', 2)可能返回25.0但后续比较会出问题
为什么不用hypot()?它更安全但有兼容性坑
hypot() 是 PHP 内置函数,专为计算直角边为 a 和 b 的斜边长度设计,等价于 sqrt(a*a + b*b),而且对大数更抗溢出。
但它在 PHP null 或字符串会警告,比如 hypot(null, 3) 触发 Warning: hypot() expects parameter 1 to be float, null given。
立即学习“PHP免费学习笔记(深入)”;
- 安全写法:
$dx = (float)$x2 - (float)$x1; $dy = (float)$y2 - (float)$y1; $distance = hypot($dx, $dy); - PHP 5.4 及以下必须自己实现,别硬用
hypot() - 注意:它只接受两个参数,不能直接塞数组或三个点
别把平面距离函数当经纬度距离用
如果实际数据是 GPS 经纬度(比如 lat: 39.9042, lng: 116.4074),直接套勾股定理会严重失真——地球是球面,1° 纬度≈111km,1° 经度随纬度变化,赤道约111km,北京约85km。
这时候该用 Haversine 公式或 geodist 类库,而不是 sqrt()。有人图省事用 sqrt(pow($lat2-$lat1,2)+pow($lng2-$lng1,2)),结果百公里误差动辄十几公里。
- 简单验证:算北京到上海(约1200km),用平面公式得出的“距离”单位是度,数值约15,完全不可比
- 若真要自己实现球面距离,至少用
deg2rad()转弧度,再套 Haversine,别跳步 - 生产环境建议直接用
geoPHP或调用Google Maps Distance Matrix API,精度和维护成本更可控
性能差异几乎可以忽略,但类型校验不能省
在普通 Web 请求里,sqrt()、pow()、hypot() 的耗时都在纳秒级,差不了多少。真正拖慢的往往是没做输入过滤:比如从数据库查出的字段是 NULL,或前端传了空字符串 '',导致计算时产生 INF 或 NAN,后续 json_encode() 直接失败。
- 务必检查变量是否为数字:
is_numeric($x1) && is_numeric($y1),注意is_numeric('1e2')返回 true,但可能不是你想要的格式 - 更稳妥用
filter_var($x1, FILTER_VALIDATE_FLOAT) !== false - 计算后建议加判断:
if (is_finite($distance)) { ... },避免INF流入前端展示
最麻烦的不是公式写错,而是把坐标当普通数字用了,却忘了它们有单位、有范围、有精度限制。











