2

我的功能有问题。我需要一张包含月份日期、日期名称和该月份工作日数量的表格。我已经在这里找到了一些帮助并对其进行了调整以适应我的需要,但是我无法运行/编译该函数这是我目前所拥有的:

create or replace TYPE DATE_ROW AS OBJECT
(
  MyDate DATE,
  Dayname VARCHAR2(12),
  Amount Integer
 )

create or replace TYPE DATE_TABLE as table of DATE_ROW

create or replace FUNCTION myfinaldays (mydate date)
       RETURN DATE_TABLE
       PIPELINED
    IS
       V_MYDATE          DATE;
       V_DAYNAME   VARCHAR2(12);
       V_AMOUNT           NUMBER;
BEGIN
    with this_month as (
        select trunc(to_date('mydate'), 'mm')+level-1 dy
          from dual 
       connect by level < (trunc(add_months(to_date('mydate'),1), 'mm')- trunc(to_date('mydate'), 'mm'))+1
    )
    FOR i IN 0 .. (select count(*) from this_month) LOOP
        select (dy) Daydate,
               to_char(dy, 'day'), Dayname,
              ( select count(*)
                  from this_month 
                 where  to_char(dy, 'dy') not in ('sat', 'sun')
               ) Amount
          from this_month
         where to_char(dy, 'dy') not in ('sat', 'sun')
           and EXTRACT(day from to_date(dy)) = i;

       pipe row (date_row(v_mydate,v_dayname, v_amount));
   END LOOP;
   RETURN;
   END;

函数调用将类似于:

select * from date_table(cast(myfinaldays('01.02.12')));

我只能给出一个日期作为参数。

我希望有人可以在这里帮助我,因为这让我慢慢发疯。任何想法、例子或想法将不胜感激。

更新:

好的,这是 mybe 的更新,对我的问题有一些更有用的信息:

这是有效的,我的目标是将它放在一个函数中,以便我可以使用 1 个参数调用它:

    with this_month as (  
    select trunc(to_date('01.02.12'), 'mm')+level-1 dy
    from   dual 
    connect by level < (trunc(add_months(to_date('01.02.12'),1), 'mm')- trunc(to_date('01.02.12'), 'mm'))+1
    )
    select (dy) mydate, (select count(*) from this_month) Days_in_month
    ,      to_char(dy, 'day') Dayname
    ,     ( select count(*) from this_month where  to_char(dy, 'dy') not in ('sat', 'sun') ) Amount
    from   this_month
    where  to_char(dy, 'dy') not in ('sat', 'sun') ;

对于我添加的循环:'and EXTRACT(day from to_date(dy))=i' 最后。

我添加了一个日期,这样你就可以看到我最后需要什么。如果我改写 mydate 并在开发人员中运行它后输入 01.02.12 作为参数,它仍然可以工作。

编译时出现的错误:-错误(10,1):PL/SQL:SQL 语句被忽略

- Error(15,5): PL/SQL: ORA-00928: Keyword SELECT missing

- Error(22,8): PLS-00113: END-Definer 'LOOP' must complete 'myfinaldays' in row 1, comlumn 10 

- Error(23,4): PLS-00103: Found the symbol "RETURN" 

由于我的 Oracle 没有运行英文,因此错误已被翻译,所以我希望猜对了。

4

2 回答 2

2

你在这里有很多问题。

  • 您不能select直接使用 a for i in 1..x(它应该是 1,而不是 0),您必须选择一个局部变量并将其用作最终值。
  • 您不能用作循环with的输入;for但你可以在那里使用光标,如下所示。
  • 循环内的选择 get no-data-found wheni表示星期六或星期日(在本例中为第四个,假设您在 2 月运行它)。
  • 获取每一行的所有非周末行的计数是低效的,尽管对于如此少量的数据并不重要。
  • 你指的是to_date('mydate');mydate 已经是一个日期,而 'mydate' 是一个与它无关的字符串。

也许还有其他一些事情,比如假设日期格式和语言环境,因为to_char(x, 'dy')不会返回satsun无处不在,但我会假设这将以受限的方式使用,所以你不会太在意。

从试图找出您期望的输出,我认为这将起作用(使用修改后的类型将days_in_month您添加到原始问题中):

create or replace type date_row as object
(
    mydate DATE,
    dayname VARCHAR2(12),
    days_in_month NUMBER,
    amount NUMBER
)
/

create or replace type date_table as table of date_row
/

create or replace function myfinaldays (mydate date)
return date_table pipelined deterministic is
begin
    for r in (
        select *
        from (
            select dy as daydate,
                to_char(dy, 'day') as dayname,
                count(*) over () as days_in_month,
                sum(case when to_char(dy, 'dy') in ('sat', 'sun')
                    then 0 else 1 end) over () as amount
            from (
                select trunc(mydate, 'mm')+level-1 dy
                from dual 
                connect by level < (trunc(add_months(mydate, 1), 'mm')
                    - trunc(mydate, 'mm')) + 1
            )
        )
        where to_char(daydate, 'dy') not in ('sat', 'sun')
    ) loop
       pipe row (date_row(r.daydate, r.dayname, r.days_in_month, r.amount));
    end loop;
end myfinaldays;
/

(如果你不想要days_in_month,那么你可以移动sat/sun检查并使用countforamount而不是sum(case). 外部select当前正在过滤星期几,因此你可以计算所有天数,但如果你不这样做,这不是问题'不想要那个值)。

然后调用为:

alter session set nls_date_format = 'DD.MM.RR';
select * from table(myfinaldays(to_date('01.02.12', 'DD.MM.RR')));

MYDATE   DAYNAME      DAYS_IN_MONTH     AMOUNT
-------- ------------ ------------- ----------
01.02.12 wednesday               29         21
02.02.12 thursday                29         21
03.02.12 friday                  29         21
06.02.12 monday                  29         21
...
29.02.12 wednesday               29         21

21 rows selected.
于 2012-09-26T12:57:45.373 回答
0

Since your function is a table valued function (returns a table), you cannot call it as regular function. you have join with it:

something like this in Oracle:

select * from date_table t
cross join table( myfinaldays(<t.date colum>)) 
于 2012-09-26T11:31:47.520 回答