- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我将 PostGIS 与 Postgresql 结合使用,以便能够通过存储在位置列 Geometry/Point SRID: 4326
中的坐标定位某个半径内的条目。这是我正在试验的两个查询:
第一个以米为单位的距离和 use_spheroid=true
EXPLAIN ANALYZE SELECT count(*) FROM "cars" WHERE ST_DWithin(location, ST_SetSRID(ST_MakePoint(20, -30), 4326), 105000, true) LIMIT 1000;
QUERY PLAN
--------------
Limit (cost=11884.28..11884.30 rows=1 width=8) (actual time=18.843..18.844 rows=1 loops=1)
-> Aggregate (cost=11884.28..11884.30 rows=1 width=8) (actual time=18.842..18.843 rows=1 loops=1)
-> Seq Scan on cars (cost=0.00..11883.33 rows=381 width=0) (actual time=0.486..18.827 rows=38 loops=1)
Filter: (((location)::geography && '0101000020E610000000000000000034400000000000003EC0'::geography) AND ('0101000020E610000000000000000034400000000000003EC0'::geography && _st_expand((location)::geography, '105000'::double precision)) AND _st_dwithin((location)::geography, '0101000020E610000000000000000034400000000000003EC0'::geography, '105000'::double precision, true))
Rows Removed by Filter: 28549
Planning time: 0.166 ms
Execution time: 18.878 ms
(7 rows)
其次,我假设接受以度为单位的距离,默认情况下 use_spheroid 为 false。更正:原来这仍然使用 use_spheroid=true,但与此调用匹配的函数签名需要几何和 SRID 单位,即 4326 的度数。
EXPLAIN ANALYZE SELECT count(*) FROM "cars" WHERE ST_DWithin(location, ST_SetSRID(ST_MakePoint(20, -30), 4326), 1) LIMIT 1000;
QUERY PLAN
-----------------------------
Limit (cost=145.30..145.31 rows=1 width=8) (actual time=0.154..0.155 rows=1 loops=1)
-> Aggregate (cost=145.30..145.31 rows=1 width=8) (actual time=0.154..0.154 rows=1 loops=1)
-> Bitmap Heap Scan on cars (cost=4.59..145.29 rows=3 width=0) (actual time=0.050..0.147 rows=37 loops=1)
Recheck Cond: (location && '0103000020E6100000010000000500000000000000000033400000000000003FC000000000000033400000000000003DC000000000000035400000000000003DC000000000000035400000000000003FC000000000000033400000000000003FC0'::geometry)
Filter: (('0101000020E610000000000000000034400000000000003EC0'::geometry && st_expand(location, '1'::double precision)) AND _st_dwithin(location, '0101000020E610000000000000000034400000000000003EC0'::geometry, '1'::double precision))
Rows Removed by Filter: 11
Heap Blocks: exact=47
-> Bitmap Index Scan on cars_location_index (cost=0.00..4.59 rows=42 width=0) (actual time=0.037..0.037 rows=48 loops=1)
Index Cond: (location && '0103000020E6100000010000000500000000000000000033400000000000003FC000000000000033400000000000003DC000000000000035400000000000003DC000000000000035400000000000003FC000000000000033400000000000003FC0'::geometry)
Planning time: 0.280 ms
Execution time: 0.188 ms
(11 rows)
两个查询返回相似的结果(+/- 因为精度)。但是第一个运行速度慢 100 倍。此外,将 use_spheroid
设置为 false 并不能保证使用索引,当距离太小 (<0.4) 或太大 (>45) 时,它会退回到 Seq Scan。这是应该的样子还是我做错了什么?
添加:经过更多实验后,我将列类型更改为 Geography.Point,现在它始终使用索引。问题似乎已解决,但我仍然对我使用 Geometry 类型观察到的行为感到困惑。
最佳答案
ST_DWithin文档指出第一个函数签名接受地理类型而不是几何类型:
boolean ST_DWithin(geography gg1, geography gg2, double precision distance_meters, boolean use_spheroid);
因为 (location,ST_SetSRID(ST_MakePoint(20, -30), 4326)) 都是几何形状,所以函数的执行很困惑是有道理的。而且我认为你的第二个函数工作正常,因为你正在执行的是这个签名:
boolean ST_DWithin(geometry g1, geometry g2, double precision distance_of_srid);
正如您所说,将列类型切换为 Geography 而不是 Geometry 将解决问题,因为这将使您正确执行:
boolean ST_DWithin(geography gg1, geography gg2, double precision distance_meters);
boolean ST_DWithin(geography gg1, geography gg2, double precision distance_meters, boolean use_spheroid);
希望对您有所帮助。
编辑:
在 documentation 中找到此部分表示在输入数据时
Standard geometry type data will autocast to geography if it is of SRID 4326
这可以解释为什么 Postgres 接受您对 ST_DWithin() 的第一次调用,因为 postgis 显然会将其转换为地理,并且这也解释了为什么执行需要更长的时间并忽略索引,因为每次转换都会产生一个新对象未在您的原始列中编入索引。
关于postgresql - ST_DWithin 有时不使用索引,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/50302239/
我将 PostGIS 与 Postgresql 结合使用,以便能够通过存储在位置列 Geometry/Point SRID: 4326 中的坐标定位某个半径内的条目。这是我正在试验的两个查询: 第一个
我在 Amazon RDS 上使用 PostreSQL 9.3 w/PostGIS 2.1.8。我有一个名为 project_location 的表,它定义了“geo-fences”(每个本质上都是一
这是我第一次创建 GIS 查询。在我的数据库的一个表中,有一列点类型。每条记录都是一台 ATM 机。我想写一个查询来获取我所在位置附近 1 公里范围内的 ATM 机。如何在 SQL 查询中使用 ST_
我有一张现有客户表和另一张潜在客户表。我想返回一个潜在客户列表,按他们出现的现有购买者的半径数排序。 每个现有客户的潜在客户表中都有许多行,并且给定现有客户周围的半径可能包含多个潜在客户。我想返回按现
我有一个包含 1756678 条记录的表 (underground_route),它定义了不同的 ug 路线。我想查找特定半径内的详细信息。 我为 the_geom 字段创建了一个 GiST 索引 u
嗨!我需要使用 GeoAlchemy 实现查询以获取给定点附近的所有点(例如,在 10 米半径内)。为了存储点,我在我的 PostGIS 数据库中使用地理字段。从 SQLAlchemy 文档中我发现我
(编辑:对不起。忘了提。Postgresql 13,PostGIS:3.1 USE_GEOS=1 USE_PROJ=1 USE_STATS=1。对此很抱歉)。 在我看来,以下 2 个查询在这两种情况下
在我的 PostgreSQL 9.3 数据库中,我有一个名为 location 的表,其中包含一个用于存储 POINT 几何的 coordinate 列。这些点是使用 SRID 4326 创建的。在某
我需要计算一个点与另一点的距离是否不超过给定的半径。我使用了 ST_DWithin 函数,在谷歌地图中,我使用两点的“这里是什么”部分获得了 lanLot。第一个:(43.2137617, 76.
我有名为 locations 的下表,其中包含以下列: 纬度 经度 现在我想查询从给定纬度/经度点开始的特定半径内的所有条目。 SELECT * FROM locations WHERE ST_DWi
我有两个线串 Line1、Line2。 line1 = "LINESTRING(72.863221 18.782499,72.863736 18.770147,72.882275 18.756169,
我发现 laravel(lumen) 查询生成器中的 postgis 函数存在问题。 流明版本:5.6 Postgres:9.6.9 与 postGIS 我有一个有效的代码: $sql = "ST_D
ST_DWithin文件说,第三个参数(距离)以米为单位。但是当我执行一些查询时,它似乎将第 3 个参数作为“度”? 这是我简化的表结构: > \d+ theuser;
我有一个包含线串的表,我想将其分成 block ,这些 block 的 ID 列表不高于为每个 block 提供的编号,并且仅存储在一定距离内的线。 比如我有一个14行的表 create table
我想获取特定位置附近的所有商店,但我的查询字符串似乎有问题。我检查过 postgis 的版本是 postgis 2.5.2_2。我还检查了经度和纬度是否具有 double 。 我的数据库结构如下:
我是一名优秀的程序员,十分优秀!