获取具有默认值的参数列表

时间:2012-12-10 14:39:35

标签: oracle oracle10g

我使用ALL_ARGUMENTS来获取oracle 10g中的参数列表,但是我找不到参数的默认值。我能做到吗?

3 个答案:

答案 0 :(得分:0)

您可能需要在10g中使用plsql编程,下面是代码示例。这种解决方案在某种意义上肯定是暴力,因为你基本上使用非常低级的原语编写函数/过程声明解析器的一部分。但是,正则表达式函数在10g中都不可用......

代码的要点是:

  • 遍历过程/函数声明的源代码行
  • 记下最近声明的例程的名称
  • 分析包含关键字“DEFAULT”的所有行,获取参数名称和默认值spec(代码示例中未详细说明)。

谨防陷阱:

  • 多行声明
  • 包含C风格评论开头字符串的C风格评论 (la / * blah blah / blah blah * * /)
  • 包含关键字的字符串文字

希望无论如何都能提供帮助,最好的问候。

代码:

DECLARE
   l_arg_and_more    VARCHAR2(400);
   l_current_unit    VARCHAR2(400);
   l_default_spec    VARCHAR2(400);
   l_offset_default  BINARY_INTEGER;
   l_offset_eoname   BINARY_INTEGER;
BEGIN
   FOR i IN (
        select text
         from all_source
        where owner = '<name of owner>'
          and name = '<object name>'
          and type in ( 'PACKAGE', 'PROCEDURE', 'FUNCTION')
          and text not like '--%'
     order by line
   ) LOOP
      IF i.text LIKE '%FUNCTION%' OR i.text LIKE '%PROCEDURE%' THEN
         IF i.text LIKE '%FUNCTION%' THEN
            l_current_unit    := LTRIM(SUBSTR(i.text, INSTR(i.text, 'FUNCTION') + LENGTH('FUNCTION')), ' ');
            l_offset_eoname   := INSTR(l_current_unit, ' ');              
            IF l_offset_eoname = 0 OR l_offset_eoname > INSTR(l_current_unit, '(') THEN
               l_offset_eoname   := INSTR(l_current_unit, '(');
            END IF;
            IF l_offset_eoname <> 0 THEN
               l_current_unit := SUBSTR(l_current_unit, 1, l_offset_eoname-1);
            ELSE
               l_current_unit := 'unidentified';
            END IF;
         END IF;
      END IF;
      --
      IF i.text LIKE '%DEFAULT%' THEN
         l_offset_default  := INSTR (i.text, 'DEFAULT');
         l_arg_and_more    := SUBSTR(i.text, 1, l_offset_default - 1);
         l_default_spec    := SUBSTR(i.text, l_offset_default + LENGTH('DEFAULT') + 1);
         --
         -- process l_arg_and_more to get the arg name, l_default_spec for the default value
         --
      END IF;
   END LOOP;              
END;

答案 1 :(得分:0)

我知道这有点太晚了,但这可能会有所帮助。我此时正面临类似的问题。需要更多测试...

SELECT SUBSTR(final_str, start_pos+1, (end_pos-start_pos-1) ) dflt_values, start_pos, end_pos  
 FROM 
 (
  SELECT start_pos, final_end_pos end_pos, MOD(ROWNUM, 2) rno, final_str  FROM
  (
   SELECT distinct start_pos, end_pos, final_str, (CASE WHEN end_pos < start_pos THEN (start_pos*2) ELSE end_pos END) final_end_pos
     FROM
    (
    SELECT Instr(final_str, '=', LEVEL) start_pos, Instr(final_str, ',', LEVEL) end_pos, final_str  --<<-- distinct 
      FROM 
    (
     SELECT RTRIM(SUBSTR(str, 1, e_start), ',')||RTRIM(SUBSTR(str, e_start, e_end-e_start), ')') final_str
     , e_start
     , e_end
 , e_end-e_start
      FROM
    (
    SELECT str, Instr(str, ',', 1, 5) e_start, Instr(str, ')') e_end
      FROM
    (
     SELECT REPLACE(REPLACE(REPLACE(SUBSTR(str, INSTR(str, '(')+1), chr(32), ''), chr(10), chr(32) ), 'DEFAULT', ':=') str
       FROM
    (
     SELECT 'PROCEDURE Some_Proc(p_arg1  VARCHAR2:= ''Arg1''
                      , p_arg2  VARCHAR2:= ''Arg2''
                      , p_arg3  DATE     DEFAULT ''SYSDATE-90''                                                                                   
                      , p_arg4  VARCHAR2 DEFAULT ''NULL''                                          
                      , p_arg5  NUMBER  := ''10'')' str
       FROM dual
    ))))
    CONNECT BY LEVEL <= LENGTH(final_str)
    )
  WHERE start_pos > 0
  ORDER BY start_pos, end_pos
 ))
 WHERE rno > 0
 /

答案 2 :(得分:0)

在视图drop-shadow中有一个名为SYS.ALL_ARGUMENTS的列,但即使在11g中,它的类型也是DEFAULT_VALUE,并且很难解决并转换为CLOB。

最简单的&#34; (不是最好的)方法是从此视图创建表格,使用LONG将此字段转换为TO_LOB并使用该字段。