从原始的trc文件中可以查询出一共包含多少条这种语句。
最近碰到一个OracleSpaital的效率问题,问题的描述具体如下:
某customer在矢量数据存储方案上选择了使用OracleSpaital的SDO_GEOMETRY进行存储,但是在前端软件的出图的速度上非常慢,已经让人不能接受了,回公司后,自己模拟了一个数据分别针对于SDO_GEOMETRY和ST_GEOMETRY两种存储方式进行了测试,发现SDO_GEOMETRY的确存在着很大的效率问题,以下为测试环境以及结果:
机器1:ArcGIS Server机器
OS: Linux Redhat 5.5+64bit
ArcGIS Server: 10+SP1
DBMS: oracle 10.2.0.4 服务器,实际上只使用了它的客户端功能
2cpu, 4core , 2.8G,4G physical memory
机器2:数据库服务器
OS: Linux Redhat 5.5+64bit
1 cpu 2core, 2.8 G ,4G physical memory
DBMS: Oracle10.2.0.4 服务器
ArcSDE: 9.3.1+sp2
数据情况:
一共80多个图层,大部分的图层没有数据,其中只有10个左右的图层有数据,一共的数据量为100M左右。
测试结果:
SDO_GEOMETRY |
ST_GEOMETRY |
|
时间 |
4.5s |
0.8s |
原因分析:
分别抓取了一下两种存储方式后台所执行的SQL语句,
原始跟踪文件为附件中的test_ora_sdo_geometry.trc和test_ora_st_geometry.trc, 格式化的trc分别为test_ora_sdo_geometry.txt和test_ora_st_geometry.txt.
从格式化后的跟踪文件中发现OracleSpatial所耗费的大部分时间花费在以下的SQL中。
SELECT DBMS_ASSERT.QUALIFIED_SQL_NAME( :1 ) FROM DUAL
call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 2508 0.04 0.03 0 0 0 0
Execute 2508 0.26 0.24 0 0 0 0
Fetch 2508 0.25 0.26 0 0 0 2508
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 7524 0.56 0.55 0 0 0 2508
Misses in library cache during parse: 1
Misses in library cache during execute: 1
Optimizer mode: ALL_ROWS
Parsing user id: 63 (SDE) (recursive depth: 1)
**************************************************************************************************
SELECT DBMS_ASSERT.SCHEMA_NAME( :1 )
FROM
DUAL
call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 1596 0.02 0.02 0 0 0 0
Execute 1596 0.16 0.15 0 0 0 0
Fetch 1596 0.16 0.16 0 0 0 1596
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 4788 0.35 0.33 0 0 0 1596
Misses in library cache during parse: 1
Misses in library cache during execute: 1
Optimizer mode: ALL_ROWS
Parsing user id: 63 (SDE) (recursive depth: 1)
****************************************************************************************************
SELECT diminfo, nvl(srid,-1)
FROM
ALL_SDO_GEOM_METADATA WHERE OWNER = :own AND TABLE_NAME = NLS_UPPER(:tab)
AND '"'||COLUMN_NAME||'"' = :col
call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 228 0.00 0.00 0 0 0 0
Execute 228 0.19 0.19 0 9 0 0
Fetch 228 1.61 1.57 0 5523 0 228
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 684 1.82 1.77 0 5532 0 228
Misses in library cache during parse: 1
Misses in library cache during execute: 1
Optimizer mode: ALL_ROWS
Parsing user id: 63 (SDE) (recursive depth: 1)
由于这些语句的(recursive depth: 1),因此都是Oracle内部所执行的SQL,在此得解释一下何为内部执行的sql,Oracle所执行的sql语句分两种,一种为用户所提交sql,另外一种为内部sql,这些内部sql语句一般都访问的是Oracle的数据字典等等,如:
在当你提交了一条select * from test语句后,Oracle后台会执行类似于select obj from obj#等查询test表的原信息的sql语句,此类语句为oracle的内部语句,其recursive depth都为大于1的值。
上述所执行的sql从字面上看一部分是后台的验证工作,如
SELECT DBMS_ASSERT.SCHEMA_NAME是验证SDE这个schema是否有问题,另一部分是用来查询表的元信息,如 SELECT diminfo, nvl(srid,-1) FROM ALL_SDO_GEOM_METADATA WHERE OWNER = :own AND TABLE_NAME = NLS_UPPER(:tab) AND '"'||COLUMN_NAME||'"' = :col。
从原始的trc文件中可以查询出一共包含多少条这种语句。
192.168.100.228 [oracle udump]$ grep DBMS_ASSERT test_ora_25346.trc|wc -l
4104
192.168.100.228 [oracle udump]$ grep "SELECT diminfo" test_ora_25346.trc|wc -l 228
而是用ST_GEOMETRY存储进行存储的时候,上述的SQL语句是不会执行的,因此大大减少了响应时间。
192.168.100.228 [oracle udump]$ grep DBMS_ASSERT test_ora_25727.trc|wc -l 0
192.168.100.228 [oracle udump]$ grep "SELECT diminfo" test_ora_25727.trc|wc -l
0
最后是同样的操作,使用两种存储方式的总的时间统计。
sql sql
SDO_GEOMETRY
OVERALL TOTALS FOR ALL NON-RECURSIVE STATEMENTS
call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 1065 0.07 0.06 0 0 0 0
Execute 1097 5.11 5.04 0 101440 456 0
Fetch 1097 0.38 0.39 0 13118 0 1055
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 3259 5.56 5.50 0 114558 456 1055
OVERALL TOTALS FOR ALL RECURSIVE STATEMENTS
call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 5848 0.17 0.14 0 0 0 0
Execute 7745 1.30 1.26 0 57 0 288
Fetch 8325 2.25 2.23 5 27905 0 8276
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 21918 3.73 3.65 5 27962 0 8564
ST_GEOMETRY
OVERALL TOTALS FOR ALL NON-RECURSIVE STATEMENTS
call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 768 0.36 0.35 0 4148 0 0
Execute 788 1.09 1.05 0 29564 0 0
Fetch 788 0.15 0.17 0 2499 0 761
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 2344 1.61 1.58 0 36211 0 761
OVERALL TOTALS FOR ALL RECURSIVE STATEMENTS
call count cpu elapsed disk query current rows
------- ------ -------- ---------- ---------- ---------- ---------- ----------
Parse 467 0.04 0.04 0 0 0 0
Execute 577 0.28 0.27 0 0 0 32
Fetch 545 0.06 0.06 0 2162 0 328
------- ------ -------- ---------- ---------- ---------- ---------- ----------
total 1589 0.39 0.37 0 2162 0 360
最终通过与Oracle中国的工程师进行沟通,Oracle的工程师确认此为Oracle产品本身的问题,需要同美国oracle研发人员沟通解决。
不要迷恋哥,哥只是个软件。