Heyho,
我必须编写一个过程,它将select-statement中的结果集插入到表中。 我的一位同事在将值从一个表复制到另一个表之前做了类似的事情。他的陈述如下:
CREATE OR REPLACE PROCEDURE Co-Worker(
pId IN INT
)
AS
BEGIN
INSERT INTO Table1_PROCESSED
SELECT * FROM Table1
WHERE ID = pId;
DELETE FROM Table1
WHERE ID = pId;
END Co-Worker;
/
这里提到的两个表具有相同的结构(实际上table1_processed只是表1的副本)。 所以我觉得“嘿!我从我的选择声明中得到了一个结果集!所以我为什么不稍微调整一下呢!” 所以我创建了这样的表:
MyTable:
TIMEID (number) | NAME (varchar2 - 128)
-----------------------------------
VALUE | VALUE
VALUE | VALUE
VALUE | VALUE
和我的程序是这样的:
CREATE OR REPLACE procedure MyProcedure(
pdate in date,
pJobtype in number default 3,
pTasktype in number default 4,
pJobstatus in number default 1,
pTaskstatus in number default 4
)
AS
pformateddate date;
BEGIN
Select to_date(to_char(to_date(pdate, 'DD.MM.YYYY HH24:MI:SS'), 'DD.MM.YYYY'), 'DD.MM.YYYY')
into pformateddate
from dual;
Insert into MyTable (TIMEID, NAME)
Select Function_GETTIMEID(to_date(st, 'DD.MM.YYYY HH24')) TIMEID
,to_char(ext) NAME
from(
Select to_char(arch_job.exec_start, 'DD.MM.YYYY HH24') st
,file.name ext
, count(file.id) cnt
from
arch_task_file
left join file on arch_task_file.File_ID = file.ID
left join arch_task on arch_task_file.Task_ID = arch_task.ID
left join arch_job on arch_task.Job_ID = arch_job.ID
where
arch_job.exec_start > pformateddate
and arch_job.exec_end <pformateddate + 1
and arch_job.jobtype_id = pJobtype
and arch_job.jobstatus_id = pJobstatus
and arch_task.Tasktype_ID = pTasktype
and arch_task.Taskstatus_ID = pTaskstatus
group by
file.name,
to_char(arch_job.exec_start, 'DD.MM.YYYY HH24'
)
);
End MyProcedure;
/
大型Select-Statement ALONE的结果如下所示:
TIMEID | NAME
-----------------------------------
VALUE | VALUE
VALUE | VALUE
VALUE | VALUE
但是如果我执行这个程序并给它一个dummydate(sysdate - 12或像'16 .07.2010 10:32:50那样的日期')我的Toad-给我一条消息“Procedure completed”我的表保持空白.. 。! 但正如之前所说的那样,大型Select-Statement会产生结果,因此不应该尝试插入空的结果集......!谁能告诉我为什么我的程序不起作用?
每个有用答案的Thx。 =)
格尔茨!
P.S .:
Select to_date(to_char(to_date(pdate, 'DD.MM.YYYY HH24:MI:SS'), 'DD.MM.YYYY'), 'DD.MM.YYYY')
into pformateddate
from dual;
需要缩短pDate值!我测试了它,所以它也可以工作,你可以在整个逻辑中忽略它。它就是为了让您全面了解情况!
答案 0 :(得分:3)
这是SQL论坛中非常常见的模式。该模式是OP说
“我在TOAD工作表中运行此SQL (或其他),它的工作原理。但当 我把它包含在不同的上下文中 - 比如存储过程 - 它 不起作用。是什么给了?“
这两个陈述不一样。在某处有一个错误的转录。也许省略了连接或添加了额外的连接。最可能的错误来源是使用存储过程中的参数替换工作表中的文字。
显然我无法告诉你差异在哪里。我所能做的就是敦促你仔细检查这两个SQL语句并找出差异。
如果您真的找不到任何差异,那么您需要调试代码。最快的方法是使用Devil的Debugger。在insert语句之后添加以下行:
dbms_output.put_line('Rows inserted = '||to_char(sql%rowcount));
您需要在TOAD中启用DBMS_OUTPUT;某处有一个标签。这至少会告诉您查询是否确实返回零行,或者您的过程是否正在插入行,而您由于某种原因没有看到它们。这是两个不同的问题。
答案 1 :(得分:0)
您需要在运行此过程的Toad会话中进行COMMIT,然后才能在任何其他会话(例如表浏览器)中查看表中的数据。你还记得那样做吗?
答案 2 :(得分:0)
与您的问题不太相关,但是
Select to_date(to_char(to_date(pdate, 'DD.MM.YYYY HH24:MI:SS'), 'DD.MM.YYYY'), 'DD.MM.YYYY')
into pformateddate
from dual;
只是一种从传递参数中删除时间元素的漫长方式。这也是做同样的事情:
Select trunc(pdate)
into pformateddate
from dual;
或者正如托尼指出的那样,一个简单的任务:
pformateddate := trunc(pdate);