[原创]如何通过临时表来加快数据库查询的性能
发表于:2007-05-26来源:作者:点击数:
标签:
下面是一些问题 需求 [quote:5d25875cb1] 由动态语句构造的游标如下: Selectfield1,field2......................... Fromtabname Wherefiled1In(1,2,3,4,5................); 用动态的方法构造如上语句,其中条件中的In部分是动态用字符串并在一起的,In的部
下面是一些问题
需求
[quote:5d25875cb1]
由动态语句构造的游标如下:
Select field1 , field2 .........................
From tabname
Where filed1 In ( 1 , 2 , 3 , 4 , 5 ................);
用动态的方法构造如上语句,其中条件中的In部分是动态用字符串并在一起的,In 的部分是一定要有的,但是这样写却是低效的,多个客户端调用执行后在缓冲池中会有多条类似的语句,浪费oracle的内存,是否有其他高效的方法可以替换一下,比如说把In后面的变量放到一个数组中(打个比方而已,Insert可以这样做,Select好像不行)以提高语句的服用性,有什么好的办法吗?
[/quote:5d25875cb1]
[quote:5d25875cb1]
我现在要处理下面的情况,大家有什么好的建议:
Oracle
数据库中有如下的表
OID NOT NULL NUMBER(3
GEOMETRY NOT NULL MDSYS.SDO_GEOMETRY
STRINGFIELD VARCHAR2(10)
DOUBLEFIELD NUMBER(25,
DOUBLEFIELD2 NUMBER(25,
BLOBFIELD BLOB
OID是主键,用户给了我一堆OID(可能很多,可能很少,不一定)
我要得到这些OID所对应的GEOMETRY对象的数据信息,当然,速度一定要快,
怎样构造这个
SQL语句呀!
请大家一定帮忙,谢谢
[/quote:5d25875cb1]
下面是
测试用例:
[code:1:5d25875cb1]
CREATE TABLE cola_markets (
mkt_id NUMBER PRIMARY KEY,
name VARCHAR2(32),
shape MDSYS.SDO_GEOMETRY);
declare
i number;
begin
i :=100;
while i<100000 loop
i:= i+1;
INSERT INTO cola_markets VALUES(
i,
'cola_a',
MDSYS.SDO_GEOMETRY(
2003, -- 2-dimensional polygon
NULL,
NULL,
MDSYS.SDO_ELEM_INFO_ARRAY(1,1003,3), -- one rectangle (1003 = exterior)
MDSYS.SDO_ORDINATE_ARRAY(1,1, i,i) -- only 2 points needed to
-- define rectangle (lower left and upper right) with
-- Cartesian-coordinate data
)
);
end loop;
exception
when no_data_found then
NULL;
end;
上面是创建cola_markets测试表的过程
从上面的表cola_markets 中得到50000个oid的数据
[/code:1:5d25875cb1]
[code:1:5d25875cb1]
declare
i number;
begin
i:= 0;
while i<50000 loop
insert into oid_tmp(OID) values(i);
i:= i+1;
end loop;
end;
[/code:1:5d25875cb1]
上面是插入表oid_tmp50000个数据的过程,下面将会用到.
这两个问题是属于同一种问题,我们该如何解决了,
下面是两种方法
我只能想到这两种方法,)
1 构造in语句
2 构造一张表,将数据放入,通过join来操作
构造in的方法,我想就不要试验了,一定是很慢的.尤其是对于50000个的,根本没有办法了.那就是剩下一个方法了,那就有几个问题需要考虑:
1 如何建立一张表,进行比较好的物理设置
2 如何快速的将50000个oid数据插入到表中
3 如何得到50000oid所对应的数据集
对于第一个问题:
如何建立一个表了,下面是对于设置一张表的物理设置:
[code:1:5d25875cb1]
initrans 4 maxtrans 255 storage ( initial 16k Next 8k minextents 1 maxextents 2147483645
pctincrease 0 freelists 4 Freelist groups 1) pctfree 10 pctused 90
[/code:1:5d25875cb1]
我只说明pctfree和pctused的含义,其他的大家自己找资料:
在oracle中对于数据的配置和操作系统的配置是一样的,数据分为数据块,
当从磁盘中读取数据的时候,内核试图先从高速缓冲中读.如果数据已经在该高速缓存中,则内核可以不必从磁盘上读.如果数据不再高速缓存中,则内核从磁盘中读数据,并将其缓存下来.
每个高速缓存数据块通过下面的方法来得到和释放
参见<unix操作系统设计>
[code:1:5d25875cb1]
算法 getblk
输入:文件系统号
块号
输出:现在能被磁盘块使用的上了锁的缓存区
{
while(没有找到缓存区)
{
if(块在散列队列中)
{
if(块忙)
{
sleep(等待"缓存区变为空闲"事件);
continue;
}
为缓存区标记为"忙";
从空闲表上摘下缓存区;
return(缓冲区);
}
else
{
if(空闲表上无缓存区)
{
sleep(等待"任何缓存区变为空闲"事件);
continue;
}
从空闲表上摘下缓存区;
if(缓冲区标记为延迟写)
{
把缓存区异步写到磁盘上;
continue;
}
从旧散列队列中摘下缓冲区;
把缓存区投到新散列队列;
return (缓冲区);
}
}
}
算法 brelse
输入:上锁态的缓存区
输出:无
{
唤醒正在等待"无论那个缓冲区变为空闲"这一事件发生的所有进程;
唤醒正在等待"这个缓冲区变为空闲"这一事件发生的所有进程;
提高处理机执行级以封锁中断;
if(缓冲区内容有效且缓冲区非"旧")
将缓存区送入空闲表尾部
else
将缓冲区送入空闲表头部
降低处理机执行级以允许中断;
给缓存区解锁;
}
[/code:1:5d25875cb1]
说明:
1 缓冲区的数据块是放入一个散列队列中的,注意是全部的数据块,不管是空闲和忙的,空闲的数据
块是放在一个freelist的双向链表中的,当内核把一个缓冲区还给缓冲区池时,它通常把该缓冲区附
加到空闲表的头部,以保证最近最少使用的原则.
2 数据块有三种状态:
1 忙 (有用户在用)
2 闲 (没有用户用)
3 延迟写 (当内核把缓冲区重新分配出去之前必须将该缓冲区内容写到磁盘上)
3 "提高处理机执行级以封锁中断;"
"降低处理机执行级以允许中断;"
的意思就是为临界区的enter和left的含义
上面说的这么多,只是让大家对"pctfree 10 pctused 90"有理解,pctfree的含义是当一个数据块中的
内容为小于10%的时候,将其放入freelist中;当一个数据块中的内容大于100%的时候,将其设置为忙,多的
空间用于数据的扩展.
对于我们这个表,当如何进行设置了,因为,这个表做的工作为要么都insert,要delete al
lrows,那我们当
将其pctfree设置为2,pctused 98,来减少io操作.
[code:1:5d25875cb1]
create table oid_tmp(oid number PRIMARY KEY)initrans 4 maxtrans 255 storage ( initial 16k \
Next 8k minextents 1 maxextents 2147483645 pctincrease 0 freelists 4 \
Freelist groups 1) pctfree 2 pctused 98 ;
[/code:1:5d25875cb1]
表设置好了,可是我们忘记了一条:这个表是一个小表,经常进行insert操作,那我们可以将其长放在内存中,
用如下的命令:
[code:1:5d25875cb1]
alter table oid_tmp storage(buffer_pool keep);
[/code:1:5d25875cb1]
通过上面的设置,我门可以看看插入50000个OID所要的时间:
[code:1:5d25875cb1]
10:41:57 SQL> declare
10:44:30 2 i number;
10:44:30 3 begin
10:44:30 4 i:=0;
10:44:30 5 while i<50000 loop
10:44:30 6 insert into oid_tmp(OID) values(i);
10:44:30 7 i:= i+1;
10:44:30 8 end loop;
10:44:30 9 end;
10:44:31 10 /
PL/SQL 过程已成功完成。
10:44:48 SQL>
[/code:1:5d25875cb1]
一共用了48-31 = 17秒钟的时间
上面都没有讲到临时表,只是用一般的表进行比较好的设置,现在,我们来看看临时表,和上面的方法进行一个比较:
创建临时表
[code:1:5d25875cb1]
Create GLOBAL temporary table oid_tmp(oid number PRIMARY KEY) on commit delete rows;
[/code:1:5d25875cb1]
[code:1:5d25875cb1]
10:49:13 SQL> declare
10:49:20 2 i number;
10:49:20 3 begin
10:49:20 4 i:= 0;
10:49:20 5 while i<50000 loop
10:49:20 6 insert into oid_tmp(OID) values(i);
10:49:20 7 i:= i+1;
10:49:20 8 end loop;
10:49:20 9 end;
10:49:21 10 /
PL/SQL 过程已成功完成。
10:49:29 SQL>
[/code:1:5d25875cb1]
一共用到了29-21 = 8秒的时间,比上面通过设置快了一倍,说明我们当用临时表来达到我们的目的;
第一个问题解决了,那我们来看看第二个问题
对于第二个问题,好像没有什么可以研究的了,不就是用上面的进行插入吗?
问题,就是我们用的oid数据都是外部的数据,不是pl/
sql的内部数据,是要通过bind到oracle数据中的,
如果向上面的这个样子,一条一条的bind,insert的话,需要很长的时间,我测试了一下,
......................140秒.
但是如果象下面的进行insert的话:
declare
begin
insert into oid_tmp(OID) values(
id1);
insert into oid_tmp(OID) values(
id2);
insert into&n&nbssp;oid_tmp(OID) values(
id3);
insert into oid_tmp(OID) values(
id4);
insert into oid_tmp(OID) values(
id5);
.....................
end
通过上面的方式来插入的话,时间为15秒,提高了9倍左右,......呵呵,这个问题解决了
第三个问题:如何得到50000oid所对应的数据集,因为有了上面的基础,就很简单了,通过
[code:1:5d25875cb1]
select * from cola_markets L oid_tmp R where L.MKT_ID = R.OID;
[/code:1:5d25875cb1]
得到记录集,根本不费时间.
说明:
对于上面的问题,我们通过临时表进行了比较好的解决,临时表还可以用到其他的方面,有时间,
我在写下来.
这几天,对于这个问题,研究了一下,不知道大家有没有什么更好的方法,望指教.
snowpy 回复于:2003-08-03 15:51:31
|
我觉得这样做法不一定比下面做法快
sFilter[50000];
for(int n = 0 ; n < 50000; ++n)
{
Bind sFilter[n] to :V0;
Select * from table where oid=:V0
}
有没有测试过?
另外我建议临时表不要使用PK,因为只有一个字段,unique scan和Full table scan应该没有太大区别吧?
|
大菠萝 回复于:2003-08-04 08:55:03
|
[quote:08c5bc21a9]
sFilter[50000];
for(int n = 0 ; n < 50000; ++n)
{
Bind sFilter[n] to :V0;
Select * from table where oid=:V0
}
[/quote:08c5bc21a9]
这没有上面快,因为厦娴氖奔涠荚谕獠勘淞康绞?菘庵械氖奔?
select没有费时间,我经过测试了的.
|
rollingpig 回复于:2003-09-28 14:12:27
|
晕
看起来很熟悉
原来是in问题
|
li2002 回复于:2003-09-29 09:27:46
|
不错,支持原创!
|
原文转自:http://www.ltesting.net
- 评论列表(网友评论仅供网友表达个人看法,并不表明本站同意其观点或证实其描述)
-