如何在不使用 Cursor 的情况下编写以下 pl/sql 块?
我在 pl/sql 块中编写了一个游标。如果该块有更多记录,则会花费大量时间。 如何在没有光标的情况下编写此内容或是否有其他替代方法可以减少时间? 是否有任何替代查询可以使用单个查询执行插入到一个表中并从另一个表中删除的操作?
DECLARE
MDLCursor SYS_REFCURSOR;
BEGIN
open MDLCursor for
select dc.dest_id, dc.digits, dc.Effectivedate, dc.expirydate
from DialCodes dc
INNER JOIN MDL d
ON dc.Dest_ID = d.Dest_ID
AND d.PriceEntity = 1
join sysmdl_calltypes s
on s.call_type_id = v_CallType_ID
and s.dest_id = dc.Dest_ID
and s.call_type_id not in
(select calltype_id from ignore_calltype_for_routing)
order by length(dc.digits) desc, dc.digits desc;
loop
fetch MDLCursor
into v_mdldest_id, v_mdldigits, v_mdlEffectiveDate, v_mdlExpDate;
insert into tt_pendingcost_temp
(Dest_ID,
Digits,
CCASDigits,
Destination,
tariff_id,
NewCost,
Effectivedate,
ExpiryDate,
previous,
Currency)
select v_mdldest_id,
Digits,
v_mdldigits,
Destination,
tariff_id,
NewCost,
Effectivedate,
ExpiryDate,
previous,
Currency
FROM tt_PendingCost
where substr(Digits, 1, 2) = substr(v_MDLDigits, 1, 2)
and instr(Digits, v_MDLDigits) = 1
and v_mdlEffectiveDate <= effectivedate
and (v_mdlExpDate > effectivedate or v_mdlExpDate is null);
if SQL%ROWCOUNT > 0 then
delete FROM tt_PendingCost
where substr(Digits, 1, 2) = substr(v_MDLDigits, 1, 2)
and instr(Digits, v_MDLDigits) = 1
and v_mdlEffectiveDate <= effectivedate
and (v_mdlExpDate > effectivedate or v_mdlExpDate is null);
end if;
exit when MDLCursor%NOTFOUND;
end loop;
close MDLCursor;
END;
I had written a cursor in a pl/sql block. This block taking lot of time if it has more records.
How to write this without a cursor or Is there any other alternative way that will reduce the time?
Is there any alternative query to perform insert into one table and delete from another table using a single query?
DECLARE
MDLCursor SYS_REFCURSOR;
BEGIN
open MDLCursor for
select dc.dest_id, dc.digits, dc.Effectivedate, dc.expirydate
from DialCodes dc
INNER JOIN MDL d
ON dc.Dest_ID = d.Dest_ID
AND d.PriceEntity = 1
join sysmdl_calltypes s
on s.call_type_id = v_CallType_ID
and s.dest_id = dc.Dest_ID
and s.call_type_id not in
(select calltype_id from ignore_calltype_for_routing)
order by length(dc.digits) desc, dc.digits desc;
loop
fetch MDLCursor
into v_mdldest_id, v_mdldigits, v_mdlEffectiveDate, v_mdlExpDate;
insert into tt_pendingcost_temp
(Dest_ID,
Digits,
CCASDigits,
Destination,
tariff_id,
NewCost,
Effectivedate,
ExpiryDate,
previous,
Currency)
select v_mdldest_id,
Digits,
v_mdldigits,
Destination,
tariff_id,
NewCost,
Effectivedate,
ExpiryDate,
previous,
Currency
FROM tt_PendingCost
where substr(Digits, 1, 2) = substr(v_MDLDigits, 1, 2)
and instr(Digits, v_MDLDigits) = 1
and v_mdlEffectiveDate <= effectivedate
and (v_mdlExpDate > effectivedate or v_mdlExpDate is null);
if SQL%ROWCOUNT > 0 then
delete FROM tt_PendingCost
where substr(Digits, 1, 2) = substr(v_MDLDigits, 1, 2)
and instr(Digits, v_MDLDigits) = 1
and v_mdlEffectiveDate <= effectivedate
and (v_mdlExpDate > effectivedate or v_mdlExpDate is null);
end if;
exit when MDLCursor%NOTFOUND;
end loop;
close MDLCursor;
END;
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(1)
我没有你的表格和数据,所以我只能猜测一些会减慢你速度的事情。
首先,游标中使用的查询中有一个
ORDER BY
子句。如果此查询返回很多行,Oracle 必须获取所有行并对它们进行排序,然后才能返回第一行。如果此查询通常返回大量结果,并且您并不特别需要它返回排序结果,那么如果您删除ORDER BY
,您可能会发现 PL/SQL 块的速度会加快一些。这样,您就可以开始从游标中获取结果,而无需获取所有结果,将它们存储在某处并首先对它们进行排序。其次,以下是
INSERT INTO ... SELECT ...
和DELETE FROM ...
语句中使用的WHERE
子句:I不明白 Oracle 如何在这些条件下有效利用索引。因此每次都必须进行全表扫描。
最后两个条件看起来很合理,但似乎没有什么可以做的。我想重点关注前两个条件,因为我认为它们还有更大的改进空间。
四个条件中的第二个是
当且仅当
Digits
以v_MDLDigits
的内容开头时,此条件成立。更好的写法是在这种情况下使用
LIKE
而不是INSTR
的优点是 Oracle 在使用LIKE
时可以利用索引>。如果您在Digits
列上有索引,Oracle 将能够在该查询中使用它。然后,Oracle 将能够专注于以v_MDLDigits
中的数字开头的那些行,而不是进行全表扫描。四个条件中的第一个是:
如果
v_MDLDigits
的长度至少为 2,并且Digits
列中的所有条目的长度也至少为 2,则此条件是多余的,因为我们之前看过的内容暗示了这一点。我不知道你为什么会出现这样的情况。我认为您可能出现这种情况的唯一原因是您在
substr(Digits, 1, 2)
上有一个功能索引。如果没有,我会很想完全删除这个substr
条件。我不认为游标是导致该过程运行缓慢的原因,并且据我所知,没有任何一条语句可以插入到一个表中并从另一个表中删除。为了使这个过程加速,我认为你只需要稍微调整一下查询。
I don't have your tables and your data so I can only guess at a couple of things that would be slowing you down.
Firstly, the query used in your cursor has an
ORDER BY
clause in it. If this query returns a lot of rows, Oracle has to fetch them all and sort them all before it can return the first row. If this query typically returns a lot of results, and you don't particularly need it to return sorted results, you may find your PL/SQL block speeds up a bit if you drop theORDER BY
. That way, you can start getting results out of the cursor without needing to fetch all the results, store them somewhere and sort them first.Secondly, the following is the
WHERE
clause used in yourINSERT INTO ... SELECT ...
andDELETE FROM ...
statements:I don't see how Oracle can make effective use of indexes with any of these conditions. It would therefore have to do a full table scan each time.
The last two conditions seem reasonable and there doesn't seem a lot that can be done with them. I'd like to focus on the first two conditions as I think there's more scope for improvement with them.
The second of the four conditions is
This condition holds if and only if
Digits
starts with the contents ofv_MDLDigits
. A better way of writing this would beThe advantage of using
LIKE
in this situation instead ofINSTR
is that Oracle can make use of indexes when usingLIKE
. If you have an index on theDigits
column, Oracle will be able to use it with this query. Oracle would then be able to focus in on those rows that start with the digits inv_MDLDigits
instead of doing a full table scan.The first of the four conditions is:
If
v_MDLDigits
has length at least 2, and all entries in theDigits
columns also have length at least 2, then this condition is redundant since it is implied by the previous one we looked at.I'm not sure why you would have a condition like this. The only reason I can think why you might have this condition is if you have a functional index on
substr(Digits, 1, 2)
. If not, I would be tempted to remove thissubstr
condition altogether.I don't think the cursor is what is making this procedure run slowly, and there's no single statement I know of that can insert into one table and delete from another. To make this procedure speed up I think you just need to tune the queries a bit.