关于WMSYS.WM_CONCAT函数的用法(oracle 10g)

问题:
今天新系统上线,发现一个包的函数的返回结果的一字段类型和测试环境不一样,导致应用程序上线后报错。
WMSYS.WM_CONCAT 函数是行转列的方法,如下面不使用函数时的查询:
SQL> Select AGENT_ID From fp_agentpolicy_dt Where POLICY_ID =140 and rownum<=5;
AGENT_ID
--------
00113
00116
00092
00097

下面的分析中是使用WMSYS.WM_CONCAT函数的结果。

分析:
测试环境(oracle 10.2.0.4):
--返回结果为varchar类型

关于WMSYS.WM_CONCAT函数的用法(oracle 10g)_第1张图片

SQL> desc wmsys.wm_concat
Parameter Type     Mode Default?
--------- -------- ---- --------
(RESULT)  VARCHAR2              
P1        VARCHAR2 IN 


生产环境(oracle 10.2.0.5):
--返回结果为clob类型
SQL> Select wmsys.wm_concat(ap.AGENT_ID) From fp_agentpolicy_dt ap
  2   WHERE ap.POLICY_ID =140 and rownum<=5;

WMSYS.WM_CONCAT(AP.AGENT_ID)
-----------------------------------
00113,00116,00092,00097


SQL> desc wmsys.wm_concat
Parameter Type     Mode Default?
--------- -------- ---- --------
(RESULT)  CLOB                  
P1        VARCHAR2 IN      

可以看出,明显是数据库的版本在做怪。

解决:
(1).在包的函数查询中用to_char进行转换后再输出结果
--在生产环境中

(2).破解移植wm_concat,使用自定义函数
wm_concat出现在oracle10g版本中,不仅是加密的,而且是在一个单独的用户中,不方便使用。并且10g以前的版本也用不上。经过破解移植,可以方便大家使用
将下边的一个type,一个函数的创建脚本执行,就可以构建自己的wm_concat(这里叫zh_concat)。
使用方法同wm_concat:

type:

create or replace TYPE zh_concat_im
AUTHID CURRENT_USER AS OBJECT
(
  CURR_STR VARCHAR2(32767),
  STATIC FUNCTION ODCIAGGREGATEINITIALIZE(SCTX IN OUT zh_concat_im) RETURN NUMBER,
  MEMBER FUNCTION ODCIAGGREGATEITERATE(SELF IN OUT zh_concat_im,
               P1 IN VARCHAR2) RETURN NUMBER,
  MEMBER FUNCTION ODCIAGGREGATETERMINATE(SELF IN zh_concat_im,
                                 RETURNVALUE OUT VARCHAR2,
                                 FLAGS IN NUMBER)
                     RETURN NUMBER,
  MEMBER FUNCTION ODCIAGGREGATEMERGE(SELF IN OUT zh_concat_im,
                    SCTX2 IN  zh_concat_im) RETURN NUMBER
);
/

create or replace TYPE BODY zh_concat_im
IS
  STATIC FUNCTION ODCIAGGREGATEINITIALIZE(SCTX IN OUT zh_concat_im)
  RETURN NUMBER
  IS
  BEGIN
    SCTX := zh_concat_im(NULL) ;
    RETURN ODCICONST.SUCCESS;
  END;
  MEMBER FUNCTION ODCIAGGREGATEITERATE(SELF IN OUT zh_concat_im,
          P1 IN VARCHAR2)
  RETURN NUMBER
  IS
  BEGIN
    IF(CURR_STR IS NOT NULL) THEN
      CURR_STR := CURR_STR || ':' || P1;
    ELSE
      CURR_STR := P1;
    END IF;
    RETURN ODCICONST.SUCCESS;
  END;
  MEMBER FUNCTION ODCIAGGREGATETERMINATE(SELF IN zh_concat_im,
                                 RETURNVALUE OUT VARCHAR2,
                                 FLAGS IN NUMBER)
    RETURN NUMBER
  IS
  BEGIN
    RETURNVALUE := CURR_STR ;
    RETURN ODCICONST.SUCCESS;
  END;
  MEMBER FUNCTION ODCIAGGREGATEMERGE(SELF IN OUT zh_concat_im,
                                   SCTX2 IN zh_concat_im)
  RETURN NUMBER
  IS
  BEGIN
    IF(SCTX2.CURR_STR IS NOT NULL) THEN
      SELF.CURR_STR := SELF.CURR_STR || ':' || SCTX2.CURR_STR ;
    END IF;
    RETURN ODCICONST.SUCCESS;
  END;
END;
/

函数:
create or replace FUNCTION zh_concat(P1 VARCHAR2)
RETURN VARCHAR2 AGGREGATE USING zh_concat_im ;
/


修改标红部门,可调整字段间分隔符

注意:经过测试,使用此方法效率会明显不如第1种方法的解决。

(3).在oracle 11可以使用LISTAGG 函数
Oracle 11g Release 2介绍了LISTAGG 函数,使得聚集连接字符串变得很容易。并且允许使用我们指定连接串中的字段顺序。使用LISTAGG如下:

COLUMN employees FORMAT A50

SELECT deptno, LISTAGG(ename, ',') WITHIN GROUP (ORDER BY ename) AS employees
FROM   emp
GROUP BY deptno;

    DEPTNO EMPLOYEES
---------- --------------------------------------------------
        10 CLARK,KING,MILLER
        20 ADAMS,FORD,JONES,SCOTT,SMITH
        30 ALLEN,BLAKE,JAMES,MARTIN,TURNER,WARD

你可能感兴趣的:(关于WMSYS.WM_CONCAT函数的用法(oracle 10g))