⼀个insert插⼊语句很慢的优化
1、insert建议
update表的时候,oracle需要⽣成redo log和undo log;此时最好的解决办法是⽤insert,并且将表设置为nologging;当把表设为nologging后,并且使⽤的insert时,速度是最快的,
这个时候oracle只会⽣成最低限度的必须的redo log,⽽没有⼀点undo信息
前提:在做insert数据之前,如果是⾮⽣产环境,请将表的索引和约束去掉,待insert完成后再建索引和约束。
1.
insert into tab1 select * from tab2;
commit;
这是最基础的insert语句,我们把tab2表中的数据insert到tab1表中。根据经验,千万级的数据可在1⼩时内完成。但是该⽅法产⽣的arch会⾮常快,需要关注归档的产⽣量,及时启动备份软件,避免arch⽬录撑爆。
2.
alter table tab1 nologging;
insert /*+ append */ into tab1 select * from tab2;
commit;
alter table tab1 logging;
该⽅法会使得产⽣arch⼤⼤减少,并且在⼀定程度上提⾼时间,根据经验,千万级的数据可在45分钟内完成。但是请注意,该⽅法适合单进程的串⾏⽅式,如果当有多个进程同时运⾏时,后发起的进程会有enqueue的等待。注意此⽅法千万不能dataguard上⽤(不过要是在database已经force logging那也是不怕的,呵呵)!!
3.
insert into tab1 select /*+ parallel */ * from tab2;
commit;
对于select之后的语句是全表扫描的情况,我们可以加parallel的hint来提⾼其并发,这⾥需要注意的是最⼤并发度受到初始化参数parallel_max_servers的限制,并发的进程可以通过v$px_session查看,
或者ps -ef |grep ora_p查看。
4.
alter session set workarea_size_policy=manual;
alter session set sort_area_size=1000000000;
alter session enable parallel dml;
insert /*+ parallel */ into tab1 select * from tab2;
commit;
与⽅法2相反,并发的insert,尚未⽐较和⽅法2哪个效率更⾼(偶估计是⽅法2快),有测试过的朋友欢迎补充。
5.
insert的固定搭配
insert into tab1 select * from tab2 partition (p1);
insert into tab1 select * from tab2 partition (p2);
insert into tab1 select * from tab2 partition (p3);
insert into tab1 select * from tab2 partition (p4);
对于分区表可以利⽤tab1进⾏多个进程的并发insert,分区越多,可以启动的进程越多。我曾经试过insert2.6亿⾏记录的⼀个表,8个分区,8个进程,如果⽤⽅法2,单个进程完成可能要40分钟,
但是由于是有8个分区8个进程,后发进程有enqueue,所以因此需要的时间为40分钟×8;但是如果⽤⽅法5,虽然单个进程需要110分钟,但是由于能够并发进程执⾏,所以总共需要的时间就约为110分钟了。
6.
DECLARE
TYPE dtarray IS TABLE OF VARCHAR2(20) INDEX BY BINARY_INTEGER;
v_col1 dtarray;
v_col2 dtarray;
v_col3 dtarray;
BEGIN
SELECT col1, col2, col3 BULK COLLECT
INTO v_col1, v_col2, v_col3
FROM tab2;
FORALL i IN 1 .. v_col1.COUNT
insert into tab1 l1 = v_col1;
END;
⽤批量绑定(bulk binding)的⽅式。当循环执⾏⼀个绑定变量的sql语句时候,在PL/SQL 和SQL引擎(engines)中,会发⽣⼤量的上下⽂切换(context switches)。使⽤bulk binding,
能将数据批量的从plsql引擎传到sql引擎,从⽽减少上下⽂切换过程,提升效率。该⽅法⽐较适合于在线处理,不必停机。
7.
sqlplus -s user/pwd<
set copycommit 2;
set arraysize 5000;
copy from user/pwd@sid -
to user/pwd@sid -
insert tab1 using select * from tab2;
exit
EOF
⽤copy的⽅法进⾏插⼊,注意此处insert没有into关键字。该⽅法的好处是可以设置copycommit和arrarysize来⼀起控制commit的频率,上⾯的⽅法是每10000⾏commit⼀次。
2、分析慢原因
使⽤10046事件分析
select*from table(dbms_xplan.display_cursor('&sqlid'));

版权声明:本站内容均来自互联网,仅供演示用,请勿用于商业和其他非法用途。如果侵犯了您的权益请与我们联系QQ:729038198,我们将在24小时内删除。