这个问题困扰了我们组的很多人。不知道这边有没有人可以提些意见帮忙解决

davidyujie 2010-01-27 08:53:20
以下存储过程从程序上看没什么问题,但是我来描述 一下问题:
1)TA723008 简称A表,TA723003_prepare简称B表。
2)A表里面有4千万条以上的数据,B表里面有1万条以上的数据,现在要对A表的某字段更新,匹配条件是A表中的日期为20071231之前的。然后就是拿A表中的某个字段去关联B表的字段,得到B表中的中文名然后更新到A表中去。
可是因为客观原因,不可能让一个存储过程跑一个晚上还在更新,因为按照以下的程序是一个笛卡尔乘积,拿一个表的全表扫描去匹配另一个表的全表扫描。
3)程序详解:逐条更新,然后通过计数器每更新1万条就commit一次。
希望有人能提出有建设性的方案来解决笛卡尔乘积 然后又能满足以上所说功能。不甚感激!!!!


PROCEDURE R_MOVE_TA723008(OI_FLAG OUT INTEGER, --0 成功, -1 失败
OS_MESG OUT VARCHAR2 --出错详细信息
) IS
VI_COUNT INTEGER := 0; --计数器
VI_COUNT_all INTEGER := 0; --计数器
CURSOR CUR_TA723008 IS
SELECT ROWID, T.*
FROM BIFT.TA723008 T
WHERE T.TA723008002 <= '20071231'
AND t.ta723008015 IS NULL;

BEGIN
OI_FLAG := 0;
OS_MESG := '更新开始';

FOR VT_TA723008 IN CUR_TA723008 LOOP
UPDATE TA723008
SET TA723008015 = (SELECT TA723003007
FROM TA723003_PREPARE
WHERE TA723003_PREPARE.TA723003005 = TA723008009
AND ROWNUM = 1)
WHERE TA723008.ROWID = VT_TA723008.ROWID;

VI_COUNT := VI_COUNT + 1;
VI_COUNT_all := VI_COUNT_all + 1;
IF (VI_COUNT >= 10000) THEN
COMMIT;
VI_COUNT := 0;
END IF;
END LOOP;
OI_FLAG := 0;
OS_MESG := '更新结束';
COMMIT;
EXCEPTION
WHEN OTHERS THEN
OI_FLAG := 1;
OS_MESG := SUBSTRB(OS_MESG || '时出错! 错误代码:' || SQLCODE || ',错误信息:' ||
SQLERRM,
1,
400);

RETURN;

END R_MOVE_TA723008;
...全文
111 8 打赏 收藏 举报
写回复
8 条回复
切换为时间正序
当前发帖距今超过3年,不再开放新的回复
发表回复
zhangwonderful 2010-02-17
  • 打赏
  • 举报
回复
循环体中的更新语句优化一下,建议少用嵌套语句,先从TA723003_PREPARE 取出TA723003007,然后再更新。

hailang1118 2010-02-17
  • 打赏
  • 举报
回复
问题应该在这句上,你查查这句执行需要多少时间。
还有把 T.*这句去掉,不知道你们这个表有多少个字段。这句很吃资源的。

CURSOR CUR_TA723008 IS
SELECT ROWID, T.*
FROM BIFT.TA723008 T
WHERE T.TA723008002 <= '20071231'
AND t.ta723008015 IS NULL;

还有看看TA723003_PREPARE这个表的TA723003005 字段有没有索引。
inthirties 2010-02-17
  • 打赏
  • 举报
回复
把表结构给出来,看看用sql能不能直接搞定。
sorry0481 2010-01-29
  • 打赏
  • 举报
回复
ACMAIN_CHM 2010-01-27
  • 打赏
  • 举报
回复
crazylaa 2010-01-27
  • 打赏
  • 举报
回复
jf
davidyujie 2010-01-27
  • 打赏
  • 举报
回复
恩 谢谢ls提醒!主要是急 所以多发了几个。不知道这种情况能否通过建立索引来达到目的
wh62592855 2010-01-27
  • 打赏
  • 举报
回复
http://topic.csdn.net/u/20100127/20/bb9500a4-19ed-44fe-a608-9b2d27795399.html?58840

请不要重复发帖 ^_^
发帖
Oracle 高级技术

3472

社区成员

Oracle 高级技术相关讨论专区
社区管理员
  • 高级技术社区
加入社区
帖子事件
创建了帖子
2010-01-27 08:53
社区公告
暂无公告