跳到主要内容

创建空间索引

seekdb 支持使用 SPATIAL 关键字创建空间索引,建表时空间索引列必须声明为 NOT NULL。支持在存储(STORED)生成列上创建空间索引,不支持在虚拟(VIRTUAL)生成列上创建。

注意事项

  • 创建空间索引的列定义中必须包含 NOT NULL 约束。
  • 创建了空间索引的列需要已经定义 SRID,否则在该列上建的空间索引在查询时无法生效。
  • 如果是在 STORED 生成列上创建空间索引,则创建列时 DDL 必须显式指定 STORED 关键字。如果在创建生成列的时候没有指定 VIRTUAL 或者 STORED 关键字,那么默认创建 VIRTUAL 生成列。
  • 在创建了索引之后,比较的时候使用列定义中的 SRID 所对应的坐标系。空间索引存储了几何对象的 MBR(Minimum Bounding Rectangle)构建,MBR 的比较方式也依赖于 SRID。

准备工作

使用 GIS 功能前,需在业务租户配置 GIS meta 数据。在连接到服务器后,执行如下命令将 default_srs_data_mysql.sql 文件导入到数据库中:

## $password 需替换为登录数据库的密码
## $localhost 需替换为登录数据库的 IP
## $tenant_name 需替换为登录数据库的租户名
## $port 需替换为数据库端口号
$python /home/admin/oceanbase/bin/import_srs_data.py -p'$password' -h $localhost -P $port -t $tenant_name -f /home/admin/oceanbase/etc/default_srs_data_mysql.sql

返回以下结果代表导入数据文件成功:

INFO: succeed to import srs data
INFO: oceanbase.__all_spatial_reference_systems old result rows -- 1
INFO: oceanbase.__all_spatial_reference_systems new result rows -- 5152

示例

创建空间索引

以下示例展示如何创建空间索引:

  • 使用 CREATE TABLE:
CREATE TABLE geom (g GEOMETRY NOT NULL SRID 4326, SPATIAL INDEX(g));
  • 使用 ALTER TABLE
CREATE TABLE geom (g GEOMETRY NOT NULL SRID 4326);
ALTER TABLE geom ADD SPATIAL INDEX(g);
  • 使用 CREATE INDEX
CREATE TABLE geom (g GEOMETRY NOT NULL SRID 4326);
CREATE SPATIAL INDEX g ON geom (g);

以下示例展示如何删除空间索引:

  • 使用 ALTER TABLE
ALTER TABLE geom DROP INDEX g;
  • 使用 DROP INDEX
DROP INDEX g ON geom;

在生成列上创建空间索引

生成列是数据库表中的一种特殊列,详细信息请参见生成列操作

以下示例展示如何在 STORED 生成列上创建空间索引:

  • 在 linestring 类型的生成列上创建空间索引,其他 POINT/POLYGON/ | MULTIPOINT/MULTILINESTRING/ | MULTIPOLYGON 类型均支持:
CREATE TABLE `receivable_items` (
`from_unit` int NOT NULL,
`to_unit` int NOT NULL,
`unit_range` linestring GENERATED ALWAYS AS (linestring(point(-(1),`from_unit`), point(1,`to_unit`))) STORED NOT NULL srid 0,
SPATIAL KEY `idx_unit_range` (`unit_range`)
);
  • 在 geometry 类型的生成列上创建空间索引:
CREATE TABLE `receivable_items` (
`id` int(32) NOT NULL auto_increment,
`geo_text` varchar(1024) NOT NULL,
`unit_range` geometry GENERATED ALWAYS AS (st_geomfromtext(geo_text, 4326, 'axis-order=long-lat')) STORED NOT NULL srid 4326
);
  • 不支持在 VIRTUAL 生成列上创建空间索引,创建语句会报错:
CREATE TABLE `receivable_items` (
`id` int(32) NOT NULL auto_increment,
`geo_text` varchar(1024) NOT NULL,
`unit_range` geometry GENERATED ALWAYS AS (st_geomfromtext(geo_text, 4326, 'axis-order=long-lat')) NOT NULL srid 4326,
SPATIAL KEY `idx_unit_range` (`unit_range`)
);
ERROR 3106 (HY000): 'unit_range' is not supported for generated columns.
  • 支持后建索引:
CREATE TABLE `receivable_items` (
`id` int(32) NOT NULL auto_increment,
`geo_text` varchar(1024) NOT NULL,
`unit_range` geometry GENERATED ALWAYS AS (st_geomfromtext(geo_text, 4326, 'axis-order=long-lat')) STORED NOT NULL srid 4326
);

INSERT INTO receivable_items(geo_text) VALUES('point(120.34904267189361 30.320965261625222)');
INSERT INTO receivable_items(geo_text) VALUES('point(120.34904267189360 30.320965261625222)');
CREATE SPATIAL INDEX IF NOT EXISTS `idx_unit_range` ON `receivable_items` (`unit_range`);
  • 支持分区表场景:
CREATE TABLE `receivable_items` (
`id` int(32) NOT NULL auto_increment primary key,
`geo_text` varchar(1024) NOT NULL,
`unit_range` geometry GENERATED ALWAYS AS (st_geomfromtext(geo_text, 4326, 'axis-order=long-lat')) STORED NOT NULL srid 4326,
SPATIAL KEY `idx_unit_range` (`unit_range`)
) PARTITION BY hash(id) PARTITIONS 3;