在MySQL 8中使用point数据类型和st_distance_sphere查找最近的位置

时间:2018-05-20 18:58:33

标签: mysql geospatial latitude-longitude mysql-8.0

我有一个名为place的表格:

id | name       | coordinates (longitude, latitude)
1  | London     | -0.12574, 51.50853
2  | Manchester | -2.25, 53.41667
3  | Glasgow    | -4.25, 55.86667

coordinates列属于点数据类型。我使用:

将点插入place
st_geomfromtext('point($longitude $latitude)', 4326)

请注意,我已经使用了SRID。

给定任何坐标,我想找到离它最近的地方(按升序排序)。我目前提出的解决方案(通过阅读MySQL文档)看起来像这样:

select
    *,
    st_distance_sphere(`place`.`coordinates`, st_geomfromtext('Point($longitude $latitude)', 4326)) as distance
from place
order by distance asc;

在这里和其他地方看到无数类似的问题之后,很明显这是一个鲜为人知的(并且更新的方式)做事,因此对此没有多少内容,因此为什么我和#39;我正在寻找一些澄清。

我的问题是:

  1. 这是最好的解决方案吗/我这样做了吗?
  2. 此方法是否会使用coordinates列上的空间索引?
  3. 使用st_distance_sphere时,是否需要指定地球的半径才能获得准确的结果? (编辑:不,它默认使用地球的半径)
  4. 编辑,以下是答案:

    explain select ...;返回:

    id | select_type | table | partitions | type | possible_keys | key  | key_len | ref  | rows   | filtered | Extra
    1  | SIMPLE      | place | NULL       | ALL  | NULL          | NULL | NULL    | NULL | 115687 | 100.00   | Using filesort
    

    flush status; select ...; show session status like 'Handler%';返回:

    Variable_name              | Value
    Handler_commit             | 1
    Handler_delete             | 0
    Handler_discover           | 0
    Handler_external_lock      | 2
    Handler_mrr_init           | 0
    Handler_prepare            | 0
    Handler_read_first         | 1
    Handler_read_key           | 1001
    Handler_read_last          | 0
    Handler_read_next          | 0
    Handler_read_prev          | 0
    Handler_read_rnd           | 1000
    Handler_read_rnd_next      | 119395
    Handler_rollback           | 0
    Handler_savepoint          | 0
    Handler_savepoint_rollback | 0
    Handler_update             | 0
    Handler_write              | 0
    

1 个答案:

答案 0 :(得分:1)

可能是最佳解决方案。让我们先得到一些其他的答案......

EXPLAIN SELECT ...说什么? (这可以回答你的Q2)。

无论其他答案如何,您的查询都将扫描整个表格。也许你最后想要LIMIT ...

另一件可能有用的事情(取决于您的应用和优化器):在WHERE子句中添加一个边界框。

在任何情况下,请执行以下操作以准确了解实际触摸的行数:

FLUSH STATUS;
SELECT ...;
SHOW SESSION STATUS LIKE 'Handler%';

回到那些答案;那么也许我们可以进一步迭代。

显示状态

好吧,Handler_read_rnd_next说它是全表扫描。 1000和1001 - 你有LIMIT 1000吗?

我推断,LIMIT并未考虑SPATIAL的工作原因。也就是说,它做了简单的事情 - (1)检查所有行,(2)排序,(3)LIMIT

那么,该怎么办?

计划A:确定您不希望结果超过X英里(km)并添加一个"边界框"查询。

B计划:放弃空间并挖掘更复杂的方式完成任务:http://mysql.rjweb.org/doc.php/latlng