Oracle 函数 wm_concat 将列转行 用法 实例

Oracle 函数 wm_concat 将列转行 用法 实例

        这篇文章主要记录下我工作中遇到sql语句查询的问题,工作中有个需求是做问卷调查,然后统计导出数据,其中有个问题是多选题,存的是答案表对应的答案的id(以逗号分隔),然后查询的时候怎么样才能把这个问题的答案也查出来拼接成逗号分隔,就用到了 WM_CONCAT 函数,下面就说一下具体用法。

      1. 首先看一下我的表结构,总共 有三张表 ,qv_question 记录了问题,qv_answer 记录了 问题对应的答案(单选,多选的选项,而不是调查回答的答案),qv_investreplay 这个记录了每个人 问卷调查 回答的 问题对应的答案,因此 一个人回答了多个问题,一个问题有多个答案,qv_investreplay 对qv_question 是一对多,同样 qv_question 对 qv_answer 也是一对多

       Oracle 函数 wm_concat 将列转行 用法 实例

Oracle 函数 wm_concat 将列转行 用法 实例

— Create table
create table QV_QUESTION
(
  id            VARCHAR2(32) not null,
  questionorder NUMBER,
  questionname  VARCHAR2(300),
  questiontype  VARCHAR2(1),
  investid      VARCHAR2(32),
  addusername   VARCHAR2(50),
  addtime       DATE,
  adddept       VARCHAR2(200),
  addname       VARCHAR2(50),
  isdeleted     VARCHAR2(50),
  investname    VARCHAR2(200),
  allowwrite    VARCHAR2(50)
)
tablespace RMSFUSION
  pctfree 10
  initrans 1
  maxtrans 255
  storage
  (
    initial 64
    minextents 1
    maxextents unlimited
  );
— Add comments to the table
comment on table QV_QUESTION
  is ‘题目表’;
— Add comments to the columns
comment on column QV_QUESTION.questionorder
  is ‘题目排序’;
comment on column QV_QUESTION.questionname
  is ‘题目名称’;
comment on column QV_QUESTION.questiontype
  is ‘类型 1 :单选2:复选  3:文本框’;
comment on column QV_QUESTION.investid
  is ‘主题id’;
comment on column QV_QUESTION.isdeleted
  is ‘0:未删除 1:删除’;
comment on column QV_QUESTION.investname
  is ‘主题名称’;
comment on column QV_QUESTION.allowwrite
  is ‘如果是单选复选的话是否允许客户手动输入 0:不允许1允许’;
— Create/Recreate primary, unique and foreign key constraints
alter table QV_QUESTION
  add constraint PK_QV_QUESTION primary key (ID)
  using index
  tablespace RMSFUSION
  pctfree 10
  initrans 2
  maxtrans 255
  storage
  (
    initial 64K
    minextents 1
    maxextents unlimited
  );
Oracle 函数 wm_concat 将列转行 用法 实例

Oracle 函数 wm_concat 将列转行 用法 实例

— Create table
create table QV_ANSWER
(
  id           VARCHAR2(32) not null,
  answername   VARCHAR2(200),
  questionid   VARCHAR2(3200),
  addusername  VARCHAR2(50),
  addtime      DATE,
  adddept      VARCHAR2(200),
  addname      VARCHAR2(50),
  isdeleted    VARCHAR2(50),
  questionname VARCHAR2(200),
  ordernumber  NUMBER
)
tablespace RMSFUSION
  pctfree 10
  initrans 1
  maxtrans 255
  storage
  (
    initial 64
    minextents 1
    maxextents unlimited
  );
— Add comments to the table
comment on table QV_ANSWER
  is ‘问题答案表’;
— Add comments to the columns
comment on column QV_ANSWER.answername
  is ‘答案名称’;
comment on column QV_ANSWER.questionid
  is ‘题目id’;
comment on column QV_ANSWER.isdeleted
  is ‘0:未删除  1:已删除’;
comment on column QV_ANSWER.questionname
  is ‘题目名称’;
comment on column QV_ANSWER.ordernumber
  is ‘答案排序’;
— Create/Recreate primary, unique and foreign key constraints
alter table QV_ANSWER
  add constraint PK_QV_ANSWER primary key (ID)
  using index
  tablespace RMSFUSION
  pctfree 10
  initrans 2
  maxtrans 255
  storage
  (
    initial 64K
    minextents 1
    maxextents unlimited
  );

Oracle 函数 wm_concat 将列转行 用法 实例

Oracle 函数 wm_concat 将列转行 用法 实例

— Create table

create table QV_INVESTREPLAY

(

  id          VARCHAR2(32) not null,

  investid    VARCHAR2(32),

  questionid  VARCHAR2(32),

  answerid    VARCHAR2(255),

  useranswer  VARCHAR2(500),

  addusername VARCHAR2(50),

  addtime     DATE,

  adddept     VARCHAR2(200),

  addname     VARCHAR2(50),

  isreplay    VARCHAR2(30),

  allowanswer VARCHAR2(255)

)

tablespace RMSFUSION

  pctfree 10

  initrans 1

  maxtrans 255

  storage

  (

    initial 64

    minextents 1

    maxextents unlimited

  );

— Add comments to the table

comment on table QV_INVESTREPLAY

  is ‘问卷反馈表’;

— Add comments to the columns

comment on column QV_INVESTREPLAY.investid

  is ‘问卷id’;

comment on column QV_INVESTREPLAY.questionid

  is ‘题目id’;

comment on column QV_INVESTREPLAY.answerid

  is ‘答案id’;

comment on column QV_INVESTREPLAY.useranswer

  is ‘用户答案(文本框)’;

comment on column QV_INVESTREPLAY.addusername

  is ‘参与者工号’;

comment on column QV_INVESTREPLAY.addtime

  is ‘参与时间’;

comment on column QV_INVESTREPLAY.adddept

  is ‘参与者部门’;

comment on column QV_INVESTREPLAY.addname

  is ‘参与者姓名’;

comment on column QV_INVESTREPLAY.isreplay

  is ‘是否已经回复 0:未回复 1:已回复’;

comment on column QV_INVESTREPLAY.allowanswer

  is ‘单选多选题客户补充答案’;

— Create/Recreate primary, unique and foreign key constraints

alter table QV_INVESTREPLAY

  add constraint PK_QV_INVESTREPLAY primary key (ID)

  using index

  tablespace RMSFUSION

  pctfree 10

  initrans 2

  maxtrans 255

  storage

  (

    initial 64K

    minextents 1

    maxextents unlimited

  );

      2. 这里只看 有个多选题, “你希望影吧播放的电影类型是?”,以及这个问题的答案。

           Oracle 函数 wm_concat 将列转行 用法 实例

           Oracle 函数 wm_concat 将列转行 用法 实例

         这是问卷调查 回答的答案,可以看到,存的 都是 qv_answer 表对应的 id,并且以 逗号分隔

            Oracle 函数 wm_concat 将列转行 用法 实例

     3. 如何用 wm_concat 函数 查出下列结果,sql 语句 如下:

        select a.id,a.addusername,a.addname,a.adddept,a.addtime,to_char(WM_CONCAT(b.answername)) 类型
                      from qv_investreplay a,qv_answer b
                              where  b.questionid = ‘422517’ and  instr(‘,’||a.answerid||’,’,’,’||b.id||’,’)>0
                                     group by a.id,a.addusername,a.addname,a.adddept,a.addtime

          Oracle 函数 wm_concat 将列转行 用法 实例

        关联条件 使用 instr(‘,’||a.answerid||’,’,’,’||b.id||’,’)>0,这样就可以关联出所有 id 对应的答案,这时候 使用 to_char(WM_CONCAT(b.answername)) 类型 将答案以逗号拼接,其他查询字段,都要在 group by 里。

        也可以用with as 做临时表 an (qv_investreplay 也可以写一个with as,只把相关字段写出来,便于测试),简单写下 如下:

              with an
                   as (
                          select id,answername from qv_answer where questionid = ‘422517’
                    )
             select a.id,a.addusername,a.addname,a.adddept,a.addtime,to_char(WM_CONCAT(b.answername)) 类型
                      from qv_investreplay a,an b
                                where instr(‘,’||a.answerid||’,’,’,’||b.id||’,’)>0
                                      group by a.id,a.addusername,a.addname,a.adddept,a.addtime

      3. 有时安装的 oracle 可能版本 并没有 wm_concat 函数,这时可以自定义,如下sql语句:

          Oracle 11gR2和12C上已经摒弃了wm_concat函数,当时我们很多程序员在程序中确使用了该函数,导致程序出现错误,为了减轻程序员修改程序的工作量,只有通过手工创建个wm_concat函数,来临时解决该问题,但是注意,及时创建了该函数,在使用的过程中,也需要用to_char(wm_concat())方式,才能完全替代之前的应用。

一.解锁wmsys用户
alter user wmsys account unlock;

二.创建包、包体和函数
以wmsys用户登录数据库,执行下面的命令

CREATE OR REPLACE TYPE WM_CONCAT_IMPL AS OBJECT
— AUTHID CURRENT_USER AS OBJECT
(
CURR_STR VARCHAR2(32767),
STATIC FUNCTION ODCIAGGREGATEINITIALIZE(SCTX IN OUT WM_CONCAT_IMPL) RETURN NUMBER,
MEMBER FUNCTION ODCIAGGREGATEITERATE(SELF IN OUT WM_CONCAT_IMPL,
P1 IN VARCHAR2) RETURN NUMBER,
MEMBER FUNCTION ODCIAGGREGATETERMINATE(SELF IN WM_CONCAT_IMPL,
RETURNVALUE OUT VARCHAR2,
FLAGS IN NUMBER)
RETURN NUMBER,
MEMBER FUNCTION ODCIAGGREGATEMERGE(SELF IN OUT WM_CONCAT_IMPL,
SCTX2 IN WM_CONCAT_IMPL) RETURN NUMBER
);
/

–定义类型body:
CREATE OR REPLACE TYPE BODY WM_CONCAT_IMPL
IS
STATIC FUNCTION ODCIAGGREGATEINITIALIZE(SCTX IN OUT WM_CONCAT_IMPL)
RETURN NUMBER
IS
BEGIN
SCTX := WM_CONCAT_IMPL(NULL) ;
RETURN ODCICONST.SUCCESS;
END;
MEMBER FUNCTION ODCIAGGREGATEITERATE(SELF IN OUT WM_CONCAT_IMPL,
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 WM_CONCAT_IMPL,
RETURNVALUE OUT VARCHAR2,
FLAGS IN NUMBER)
RETURN NUMBER
IS
BEGIN
RETURNVALUE := CURR_STR ;
RETURN ODCICONST.SUCCESS;
END;
MEMBER FUNCTION ODCIAGGREGATEMERGE(SELF IN OUT WM_CONCAT_IMPL,
SCTX2 IN WM_CONCAT_IMPL)
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 wm_concat(P1 VARCHAR2)
RETURN VARCHAR2 AGGREGATE USING WM_CONCAT_IMPL ;
/

三.创建同义词并授权

create public synonym WM_CONCAT_IMPL for wmsys.WM_CONCAT_IMPL
/
create public synonym wm_concat for wmsys.wm_concat
/

grant execute on WM_CONCAT_IMPL to public
/
grant execute on wm_concat to public
/

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/111327.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • sudo chmod 755 ….指令分析

    sudo chmod 755 ….指令分析sudo:使用管理员root权限执行指令。chmod:文件调用权限分为三级:文件拥有者、群组、其他。利用chmod可以改变文件权限。775:7,7,5各代表一个权限其中a,b,c各为一个数字,分别表示User、Group、及Other的权限。-rwx-r–r–(一共10个参数)表示文件所属组和用户的对应权限。第一个跟参数属于管理员,跟chmod无关,先不管.2-4…

  • oracle错误代码大全(超详细)

    oracle错误代码大全(超详细)本篇文章是对oracle错误代码进行了详细的总结与分析,需要的朋友参考下ORA-00001:违反唯一约束条件(.)ORA-00017:请求会话以设置跟踪事件ORA-00018:超出最大会话数ORA-00019:超出最大会话许可数ORA-00020:超出最大进程数()ORA-00021:会话附属于其它某些进程;无法转换会话ORA-00022:无效的会话ID;访问被拒绝ORA-00023:会话引用进程私用内存;无法分离会话ORA-00024:单一进程模式下不允许从多个

  • samba服务共享目录时 什么参数表示该共享目录可以浏览_电脑samba共享

    samba服务共享目录时 什么参数表示该共享目录可以浏览_电脑samba共享会不会有那么一天,生活可以简单到每天清早踏上一辆载着鲜花的脚踏车,微笑着穿过窄窄的街巷,为爱花的人送去芬芳,为需要知识的你送去帮助。上期为大家说了如何搭建dhcp服务(详情点击使用DHCP动态分配IP)本期为大家带来搭建samba服务先来说说samba服务的作用:跨平台支持文件共享服务samba的特点:支持匿名和身份验证共享数据,它的安全性也比较高samba支持的协议有:nmb(监听137,138号端口,提供域名访问,属于UDP协议)smb(监听139号端口,服务消息块,提供Linux平台共享

  • SQL Server数据库分区分表

    SQL Server数据库分区分表当一个数据表的数据量达到千万级别以后,每次查询都需要消耗大量的时间,所以当表数据量达到一定量级后我们需要对数据表水平切割。水平分区分表就是把逻辑上的一个表,在物理上按照你指定的规则分放到不同的文件里,把一个大的数据文件拆分为多个小文件,还可以把这些小文件放在不同的磁盘下。这样把一个大的文件拆分成多个小文件,便于我们对数据的管理。下面我们来创建表分区代码创建分区表添加文件组代码格式:…

  • CreateThread用法详解[通俗易懂]

    CreateThread用法详解[通俗易懂]CreateThread用法详解今天我给大家讲一讲C++中的多线程编程技术,C++本身并没有提供任何多线程机制,但是在windows下,我们可以调用SDK win32 api来编写多线程的程序,下面我就此简单的讲一下:创建线程的函数  HANDLE CreateThread(     LPSECURITY_ATTRIBUTES lpThreadAttributes, //

  • java中decode和encode_java encoding

    java中decode和encode_java encoding一概述前端传递过来的url被转码了;应该chuan

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号