ST_GEOMETRY
ST_GEOMETRY这种存储方式是从ArcSDE9.2开始支持的,但是9.2还是将SDELOB作为默认的存储方式,并没有将其作为默认的存储方式,直到ArcSDE后才开始将其作为默认的存储方式。
在详细介绍ST_GEOMETRY之前先介绍一下Oracle的复合数据类型,从Oracle 9开始,Oracle除了支持常规数据类型,如NUMBER,VARCHAR等,还开始支持复合数据类型,在Oracle中创建复合数据类型非常类似于在C++中创建类,只不过指定这个规则的是Oracle而已,我们自己可以随意的创建复合数据类型,如下面我自己创建了一个Person 的复合数据类型,这个数据类型可以存储个人信息:
CREATE TYPE person_typ AS OBJECT (
idno NUMBER,
first_name VARCHAR2(20),
last_name VARCHAR2(25),
email VARCHAR2(25),
phone VARCHAR2(20),
MAP MEMBER FUNCTION get_idno RETURN NUMBER,
MEMBER PROCEDURE display_details ( SELF IN OUT NOCOPY person_typ ));
CREATE TYPE BODY person_typ AS MAP
MEMBER FUNCTION get_idno RETURN NUMBER IS BEGIN RETURN idno;
END;
MEMBER PROCEDURE display_details ( SELF IN OUT NOCOPY person_typ ) IS BEGIN
-- use the PUT_LINE procedure of the DBMS_OUTPUT package to display details
DBMS_OUTPUT.PUT_LINE(TO_CHAR(idno) || ' ' || first_name || ' '|| last_name); DBMS_OUTPUT.PUT_LINE(email || ' ' || phone);END;
END
;
/
从上面的定义上,有成员变量,有成员方法,其格式非常类似于C++中的类。
然后可以创建一个表,其某个字段的类型可以选择person_typ,作为存储类型,如:
CREATE TABLE contacts (
contact person_typ,
contact_date DATE );
INSERT INTO contacts VALUES (
person_typ (65, 'Verna', 'Mills', '[email protected]', '1-800-555-4412'),
'24 Jun 2003' );
ST_GEOMETRY也是一个复合存储类型,不同的是它是由ESRI公司自己定义的,用来存储空间数据的复合存储类型,它的定义也是透明的,可以使用下面的命令进行查看:
SQL> desc sde.st_geometry
sde.st_geometry is
NOT FINAL
Name Null? Type
----------------------------------------- -------- ---------------------------
ENTITY NUMBER(38)
NUMPTS NUMBER(38)
MINX FLOAT(64)
MINY FLOAT(64)
MAXX FLOAT(64)
MAXY FLOAT(64)
MINZ FLOAT(64)
MAXZ FLOAT(64)
MINM FLOAT(64)
MAXM FLOAT(64)
AREA FLOAT(64)
LEN FLOAT(64)
SRID NUMBER(38)
POINTS BLOB
METHOD
------
FINAL CONSTRUCTOR FUNCTION ST_GEOMETRY RETURNS SELF AS RESULT
Argument Name Type In/Out Default?
------------------------------ ----------------------- ------ --------
GEOM_STR CLOB IN
SRID NUMBER IN
METHOD
------
MEMBER FUNCTION ST_AREA RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_LEN RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_LENGTH RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_ENTITY RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_NUMPTS RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MINX RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MAXX RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MINY RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MAXY RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MINM RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MAXM RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MINZ RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_MAXZ RETURNS NUMBER
METHOD
------
MEMBER FUNCTION ST_SRID RETURNS NUMBER
METHOD
------
STATIC FUNCTION GET_RELEASE RETURNS NUMBER
METHOD
------
FINAL CONSTRUCTOR FUNCTION ST_GEOMETRY RETURNS SELF AS RESULT
Argument Name Type In/Out Default?
------------------------------ ----------------------- ------ --------
X NUMBER IN
Y NUMBER IN
Z NUMBER IN
M NUMBER IN
SRID NUMBER IN
其中红色部分为该复合数据类型的成员变量,其中比较重要的环境变量为POINTS,它的类型为BLOB,该成员变量就是存储空间点串坐标的变量。
同样我们可以通过各种工具来创建包含该数据类型字段的表,如下面是我们用ArcCatalog所创建的FeatureClass cities的表结构:
Oracle数据库:
SQL> desc cities_2
Name Null Type
----------------------------------------- -------- ----------------------------
OBJECTID NOT NULL NUMBER(38)
CITY_NAME NVARCHAR2(30)
GMI_ADMIN NVARCHAR2(7)
ADMIN_NAME NVARCHAR2(42)
FIPS_CNTRY NVARCHAR2(2)
CNTRY_NAME NVARCHAR2(30)
STATUS NVARCHAR2(50)
POP_RANK NUMBER(5)
POP_CLASS NVARCHAR2(22)
PORT_ID NUMBER(5)
LABEL_FLAG NUMBER(5)
SHAPE ST_GEOMETRY
DB2数据库:
[sde@testserver ~]$ db2 describe table sde.test
type name Length Scale Nulls
------------------------------- --------- ------------------- ---------- ----- ------
OBJECTID SYSIBM INTEGER 4 0 No
NAME SYSIBM VARCHAR 256 0 Yes
SHAPE DB2GSE ST_MULTIPOLYGON 0 0 Yes
RASTER SYSIBM INTEGER 4 0 Yes
从该表的表结构可以看出,ST_GEOMETRY相对于的SDELOB存储方式已经将B表和F表合而为一了,以前的F表已经作为B表的一个SHAPE字段存在于B表中。
然后再介绍一下ST_GEOMETRY的索引结构:
ST_GEOMETRY存储方式的空间索引实现方式是在数据库层次上实现的,并不是在应用层次上实现的,之所以会这么说,在下面会有相应的介绍。
先介绍一下存储索引信息的位置:
和SDELOB存储方式一样,ST_GEOMETRY的具体的索引信息也是存储S表,该S表的表结构和SDELOB存储的方式一样,在此不重复介绍。
和SDELOB存储方式不同的是,它在数据库中是存在着一个相应的空间索引的定义,我将定义两个字标红的意思是该索引只是定义的一个数据库对象,其内部并不存储任何的数据,真是的索引信息是存储在S表中的,下面的数据为Cities_2表中所有的所有的索引信息:
SQL> select index_name,index_type from user_indexes where table_name='CITIES_2';
INDEX_NAME INDEX_TYPE
------------------------------ ---------------------------
SYS_IL0000092702C00027$$ LOB
R2068_SDE_ROWID_UK NORMAL
A1465_IX1 DOMAIN
在列出的索引中多了一种类型为DOMAIN的索引,该索引为oracle的域索引,实际上就是自定义索引。我们接着查看一下该域索引的定义:
SQL> select dbms_metadata.get_ddl('INDEX','A1465_IX1','SDE') from dual;
DBMS_METADATA.GET_DDL('INDEX','A1465_IX1','SDE')
--------------------------------------------------------------------------------
CREATE INDEX "SDE"."A1465_IX1" ON "SDE"."CITIES_2" ("SHAPE") INDEXTYPE IS "SDE"."ST_SPATIAL_INDEX" PARAMETERS ('ST_GRIDS = 4.3144346983011
T_COMMIT_ROWS = 10000 PCTFREE 0 INITRANS 4')
其中红色位置标出来的是需要注意的,
SHAPE:表示是在SHAPE字段上所建立的索引。
INDEXTYPE:索引的类型,实际上这是一个最关键的对象,在下面进行更详细的介绍
ST_GRIDS:各级网格的大小,因为该图层上我只建立了一级索引,因此在这个例子上只包括一级索引
下面接着看看SDE.ST_SPATIAL_INDEX这个对象在oracle中是如何定义的:
SQL> select dbms_metadata.get_ddl('INDEXTYPE','ST_SPATIAL_INDEX','SDE') from dual; DBMS_METADATA.GET_DDL('INDEXTYPE','ST_SPATIAL_INDEX','SDE') -------------------------------------------------------------------------------- CREATE OR REPLACE INDEXTYPE "SDE"."ST_SPATIAL_INDEX" FOR "SDE"."ST_BUFFER_INTERSECTS" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY", NUMBER),
"SDE"."ST_CROSSES" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY"), "SDE"."ST_ENVINTERSECTS" ("SDE"."ST_GEOMETRY", NUMBER, NUMBER, NUMBER, NUMBER),
"SDE"."ST_ENVINTERSECTS" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY"), "SDE"."ST_ENVINTERSECTS" ("SDE"."ST_GEOMETRY", NUMBER, NUMBER, NUMBER, NUMBER,VARCHAR2),
"SDE"."ST_ENVINTERSECTS" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY", VARCHAR2),
"SDE"."ST_EQUALS" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY"), "SDE"."ST_INTERSECTS" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY"), "SDE"."ST_ORDERINGEQUALS" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY"), DBMS_METADATA.GET_DDL('INDEXTYPE','ST_SPATIAL_INDEX','SDE') -------------------------------------------------------------------------------- "SDE"."ST_OVERLAPS" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY"), "SDE"."ST_RELATE" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY", VARCHAR2), "SDE"."ST_TOUCHES" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY"), "SDE"."ST_WITHIN" ("SDE"."ST_GEOMETRY", "SDE"."ST_GEOMETRY") USING "SDE"."ST_DOMAIN_METHODS" WITH LOCAL RANGE PARTITION
其中红色部分为该对象所支持所有的Operator对象,只有列在这里面的操作符对象才有可能用上索引。
绿色部分为实现网格空间索引算法的包。
还剩下最后一个问题,就是为什么说ST_GEOMETRY的空间索引是在数据库层次上实现的而不是在应用层次上实现的。大比例尺下浏览数据的时候,抓取一下后台一下相应的SQL语句如下:
*** 2010-12-16 09:41:09.632
*** ACTION
NAME:() 2010-12-16 09:41:09.631
*** MODULE
NAME:(ArcCatalog.exe) 2010-12-16 09:41:09.631
*** SERVICE NAME:(SYS$USERS) 2010-12-16 09:41:09.631
*** SESSION ID:(152.4487) 2010-12-16 09:41:09.631
=====================
PARSING IN CURSOR #25 len=69 dep=0 uid=68 oct=3 lid=68 tim=1262171552374697 hv=2558560494 ad='8ca78d78'
SELECT table_name, time_last_modified FROM SDE.sde_tables_modified
END OF STMT
EXEC #25:c=0,e=44,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,tim=1262171552374688
FETCH #25:c=0,e=52,p=0,cr=1,cu=0,mis=0,r=7,dep=0,og=1,tim=1262171552375294
=====================
PARSING IN CURSOR #46 len=288 dep=0 uid=68 oct=3 lid=68 tim=1262171552380038 hv=1466667998 ad='806b7fd8'
SELECT 1 SHAPE, CITIES_2.OBJECTID, CITIES_2.SHAPE.points,CITIES_2.SHAPE.numpts,CITIES_2.SHAPE.entity,CITIES_2.SHAPE.minx,CITIES_2.SHAPE.miny,CITIES_2.SHAPE.maxx,CITIES_2.SHAPE.maxy,CITIES_2.rowid FROM SDE.CITIES_2 CITIES_2 WHERE SDE.ST_EnvIntersects(CITIES_2.SHAPE,:1,:2,:3,:4) = 1
END OF STMT
EXEC #46:c=1000,e=1162,p=0,cr=0,cu=0,mis=0,r=0,dep=0,og=1,tim=1262171552380029
=====================
PARSING IN CURSOR #48 len=158 dep=1 uid=68 oct=3 lid=68 tim=1262171552380494 hv=4131701150 ad='806b5468'
SELECT distinct sp_id FROM SDE.S258_IDX$ WHERE gx >= :1 AND gx <= :2 AND gy >= :3 AND gy <= :4 AND minx <= :e1 AND miny <= :e2 AND maxx >= :e3 AND maxy >= :e4
END OF STMT
EXEC #48:c=0,e=117,p=0,cr=0,cu=0,mis=0,r=0,dep=1,og=1,tim=1262171552380485
FETCH #48:c=3000,e=2679,p=0,cr=18,cu=0,mis=0,r=100,dep=1,og=1,tim=1262171552383304
FETCH #48:c=0,e=177,p=0,cr=0,cu=0,mis=0,r=100,dep=1,og=1,tim=1262171552383632
FETCH #48:c=0,e=185,p=0,cr=0,cu=0,mis=0,r=100,dep=1,og=1,tim=1262171552383946
FETCH #48:c=999,e=189,p=0,cr=0,cu=0,mis=0,r=100,dep=1,og=1,tim=1262171552384262
FETCH #48:c=0,e=184,p=0,cr=0,cu=0,mis=0,r=100,dep=1,og=1,tim=1262171552384574
从上面所执行的SQL中可以看出的确访问了S258_IDX$表,更重要的是和SDELOB存储方式不同的是dep=1,这说明这条SQL并不是由前端应用所提交给数据库,而是数据库自己所执行的,换句话说是因为执行,前一个SQL:
SELECT 1 SHAPE, CITIES_2.OBJECTID, CITIES_2.SHAPE.points,CITIES_2.SHAPE.numpts,CITIES_2.SHAPE.entity,CITIES_2.SHAPE.minx,CITIES_2.SHAPE.miny,CITIES_2.
SHAPE.maxx,CITIES_2.SHAPE.maxy,CITIES_2.rowid FROM SDE.CITIES_2 CITIES_2 WHERE SDE.ST_EnvIntersects(CITIES_2.SHAPE,:1,:2,:3,:4) = 1
而导致Oracle所执行的SQL,所以ST_GEOMETRY的空间索引实现是在数据库层次上实现的。