ORA-06502:PL / SQL:数字或值错误:使用Oracle聚合函数时字符串缓冲区太小

时间:2010-06-09 11:14:23

标签: oracle plsql aggregate-functions

美好的一天大师,

我有一个脚本定期填充表崩溃并给出上述错误。奇怪的是,它已经在生产系统上运行了近3个月,没有任何问题,上周突然崩溃了。据我所知,这些表格没有任何变化。

以前有没有人遇到这样的事情?我相信它与我在其中实现的聚合函数有关;但它最初起作用了。

请;请将我开发的脚本部分附加到我认为给出错误的过程中。

创建或替换程序V1   IS

- DECLARE

    v_a       VARCHAR2(4000);
    v_b       VARCHAR2(4000);
    v_c       VARCHAR2(4000);
    v_d       VARCHAR2(4000);
    v_e       VARCHAR2(4000);
    v_f       VARCHAR2(4000);
    v_g       VARCHAR2(4000);
    v_h       VARCHAR2(4000);
    v_i       VARCHAR2(4000);
    v_j       VARCHAR2(4000);
    v_k       VARCHAR2(4000);
    v_l       VARCHAR2(4000);
    v_m       VARCHAR2(4000);
    v_n       NUMBER(10);
    v_o       VARCHAR2(4000);

-
       - 填充DEMO表的程序

        BEGIN

              -- Delete all from the DEMO table
              DELETE FROM DEMO;

              -- Populate fields in DEMO from DEMOV1
              INSERT INTO DEMO(ID, D_ID, CTR_ID, C_ID, DT_NAM, TP, BYR, ENY,
                               ONG, SUMM, DTW, REV, LD, MD, STAT, CRD)
              SELECT ID, D_ID, CTR_ID, C_ID, DT_NAM, TP, TO_NUMBER(TO_CHAR(BYR,'YYYY')), 
                   TO_NUMBER(TO_CHAR(NVL(ENY,SYSDATE),'YYYY')), CASE WHEN ENY IS NULL THEN 'Y' ELSE 'N' END, SUMMARY, DTW,
                   REV, LD, MD, '1', SYSDATE 
              FROM DEMOV1;

          -- LOOP THROUGH DEMO TABLE
          FOR j IN (SELECT ID, CTR_ID, C_ID FROM DEMO)
          LOOP




                Select semic_concat(TXTDESC)
                INTO v_a 
                From GEOT
                WHERE ID = j.ID;



               SELECT COUNT(*)
               INTO v_n
               FROM MERP M, PROJ P
               WHERE M.MID = P.COD
               AND ID = j.ID
               AND PROAC IS NULL;

               IF (v_n > 0)
               THEN


                    Select semic_concat(PRO)
                    INTO v_b 
                    FROM MERP M, PROJ P
                    WHERE M.MID = P.COD
                    AND ID = j.ID;

               ELSE


                    Select semic_concat(PRO || '(' || PROAC || ')' )
                    INTO v_b 
                    FROM MERP M, PROJ P
                    WHERE M.MID = P.COD
                    AND ID = j.ID;

               END IF;


                Select semic_concat(VOCNAME('P02',COD))
                INTO v_c 
                From PAR
                WHERE ID = j.ID;



                Select semic_concat(VOCNAME('L05',COD))
                INTO v_d 
                From INST
                WHERE ID = j.ID;


                Select semic_concat(NVL(AUTHOR,'Anon') ||' ('||to_char(PUB,'YYYY')||') '||TITLE||', '||EDT)
                INTO v_e 
                From REFE
                WHERE ID = j.ID;



                Select semic_concat(NAM)
                INTO v_f 
                FROM EDM E, EDO EO
                WHERE E.EDMID = EO.EDOID
                AND ID = j.ID;



                Select semic_concat(VOCNAME('L08', COD))
                INTO v_g 
                FROM AVA 
                WHERE ID = j.ID;



               SELECT or_concat(NAM)
               INTO v_o
               FROM CON 
               WHERE ID = j.ID
               AND NAM = 'Unknown';

                    IF (v_o = 'Unknown')
                    THEN

                        Select or_concat(JOBTITLE || ' (' || EMAIL || ')')
                        INTO v_h 
                        FROM CON 
                        WHERE ID = j.ID;

                    ELSE

                        Select or_concat(NAM || ' (' || EMAIL || ')')
                        INTO v_h 
                        FROM CON 
                        WHERE ID = j.ID;

                    END IF;



                Select commaencap_concat(COD)
                INTO v_i 
                FROM PAR 
                WHERE ID = j.ID;

                IF (v_i = ',')
                THEN

                    v_i := null;

                ELSE

                    Select commaencap_concat(COD)
                    INTO v_i 
                    FROM PAR 
                    WHERE ID = j.ID;


                END IF;



                Select commaencap_concat(COD)
                INTO v_j 
                FROM INST 
                WHERE ID = j.ID;

                IF (v_j = ',')
                THEN

                    v_j := null;

                ELSE

                    Select commaencap_concat(COD)
                    INTO v_j 
                    FROM INST
                    WHERE ID = j.ID;

                END IF;



                Select commaencap_concat(COD)
                    INTO v_k 
                    FROM SAR 
                    WHERE ID = j.ID;

                IF (v_k = ',')
                THEN

                    v_k := null;

                ELSE

                    Select commaencap_concat(COD)
                    INTO v_k 
                    FROM SAR 
                    WHERE ID = j.ID;

                END IF;




                Select commaencap_concat(CONID)
                    INTO v_l 
                    FROM CON 
                    WHERE ID = j.ID;

                IF (v_l = ',')
                THEN

                    v_l := null;

                ELSE

                    Select commaencap_concat(CONID)
                    INTO v_l 
                    FROM CON 
                    WHERE ID = j.ID;

                END IF;



                Select commaencap_concat(PROID)
                    INTO v_m 
                    FROM PRO 
                    WHERE ID = j.ID;

                IF (v_m = ',')
                THEN

                    v_m := null;

                ELSE

                    Select commaencap_concat(PROID)
                    INTO v_m 
                    FROM PRO 
                    WHERE ID = j.ID;

                END IF;

                 -- UPDATE DEMO TABLE
                 UPDATE DEMO
                 SET GEOC = v_a,
                     PRO = v_b,
                     PAR = v_c,
                     INS = v_d,
                     REFER = v_e,
                     ORGR = v_f,
                     AVAY = v_g,
                     CON = v_h,
                     DTH = v_i,
                     INST = v_j,
                     SA = v_k,
                     CC = v_l,
                     EDPR = v_m,
                     CTR = (SELECT NAM
                                  FROM EDM
                                  WHERE EDMID = j.CTR_ID),
                     COLL = (SELECT NAM
                                    FROM EDM
                                    WHERE EDMID = j.C_ID)

                WHERE ID = j.ID;

          END LOOP;


        END V1;

/

聚合函数,commaencap_concat(用逗号封装),or_concat(带或的concats)和semic_concat(带分号的concats)。

使用的其余表格都链接到主表DEMO。

我检查了列大小,似乎没有问题。我尝试单独执行SELECT语句,并且在不填充表的情况下给出相同的错误。

任何线索?

非常感谢您的预期支持。

感谢APC的所有帮助;特别是rowtype和debug建议。我已经检查了所有列,并且已经将那些小于4000字节的列调整为4000字节,但我仍然收到相同的错误消息。

我尝试的SELECT语句是使用过程中的一个聚合函数:

            Select semic_concat(TXTDESC)
           -- INTO v_a 
            From GEOT
            WHERE ID IN (SELECT ID FROM DEMO);

它引发了字符串缓冲区的相同错误太小;但当我在声明中添加了一个小组时,它起作用了。

            Select semic_concat(TXTDESC)
            -- INTO v_a 
            From GEOT
            WHERE ID IN (SELECT ID FROM DEMO)
            GROUP BY ID;

我尝试在过程中交换所有出现的此类语句,但仍然会出现相同的错误。再次感谢;仍在努力。

1 个答案:

答案 0 :(得分:5)

聚合函数,呃,聚合。这意味着他们添加数字或将字符串连接在一起。源值越大,产品越大。

在不知道更多细节的情况下很难确定,但最可能的解释是数据中的某些内容发生了变化,结果是您的聚合值现在太大而无法容纳目标栏。

修改

  

“我在其上尝试了SELECT语句   拥有并给出同样的错误......“

嗯,哪个SELECT?你的程序很多。

无论如何,您需要做的是学习调试代码。的(1)

调试PL / SQL的最佳方法是使用支持此类工作的IDE。 TOAD和PL / SQL Developer都这样做,Oracle自己的(免费)产品SQL Developer也是如此。 Find out more

或者你可以使用DBMS_OUTPUT(AKA the Devil's Debugger)并插入大量的DBMS_OUTPUT.PUT_LINE()调用来查看你将要执行的语句,以及相关值的长度。

但是,假设所有PL / SQL字符串变量都是SQL列的最大长度 - varchar2(4000) - 我会专注于填充v_n变量的代码。 number(10)绝不是SQL中可容纳的最大数字,因此您违反了最有可能成为候选者的缓冲区限制。虽然,由于错误消息确实提到 string 缓冲区,我可能会卖给你一个流浪汉。

另一种可能性是最终的UPDATE语句。 DEMO的所有列都是varchar2(4000)吗?如果没有,那么你需要看看它们。最好使用%TYPE语法指定变量:

 v_a demo.geoc%TYPE;
 v_b demo.pro%TYPE;

或者,要保存输入,请指定一个记录变量:

v_demo demo%rowtype;

您可以这样引用:

select semic_concat(TXTDESC) 
into v_demo.geoc  
From GEOT 
WHERE ID = j.ID; 

(顺便说一句,可以使用UPDATE ... SET ROW =语法在更新语句中使用行级变量,但我认为这不适合您的情况。)

编辑2

同样,NO_DATA_FOUND指向数据问题。除非我们的数据库是只读的,否则我们必须预期数据会发生变化,我们应该处理与数据相关的异常。如果您不处理NO_DATA_FOUND的原因是因为数据应始终存在,则您有更广泛的问题,可能是丢失或禁用的外键。通常,假设我们将获得NO_DATA_FOUNDTOO_MANY_ROWS等并包含有用的异常处理程序来记录相关详细信息更安全。


脚注(1)或者学会开发测试首先使用单元测试工具,但这是一个稳定的门,马走了场景......