将逗号分隔值拆分为Oracle中的列

时间:2015-07-16 20:50:45

标签: sql oracle split

我返回的值为255个逗号分隔值。有没有一种简单的方法可以将它们分成不包含255个子列的列?

ROW  | VAL
----------- 
1    | 1.25, 3.87, 2, ... 
2    | 5, 4, 3.3, ....

ROW | VAL | VAL | VAL ...
--------------------- 
1   |1.25 |3.87 | 2 ...     
2   | 5   | 4   | 3.3 ...

4 个答案:

答案 0 :(得分:18)

小心!如果列表中有null元素并且您希望该项目或其后的项目,则格式'[^,]+'的regexp_substr表达式将不会返回预期值。考虑这个例子,其中第4个元素是NULL,我想要第5个元素,因此期望' 5'退回:

SQL> select regexp_substr('1,2,3,,5,6', '[^,]+', 1, 5) from dual;

R
-
6

惊喜!它返回第5个NON-NULL元素,而不是实际的第5个元素!返回的数据不正确,您甚至可能无法捕获它。试试这个:

SQL> select regexp_substr('1,2,3,,5,6', '(.*?)(,|$)', 1, 5, NULL, 1) from dual;

R
-
5

因此,上面更正的REGEXP_SUBSTR表示要查找第5次出现的0个或更多逗号分隔字符,后跟逗号或行尾(允许下一个分隔符,可以是逗号或结尾)并且当找到时返回第一个子组(数据不包括逗号或行尾)。

搜索匹配模式'(.*?)(,|$)'解释:

(             = Start a group
.             = match any character
*             = 0 or more matches of the preceding character
?             = Match 0 or 1 occurrences of the preceding pattern
)             = End the 1st group
(             = Start a new group (also used for logical OR)
,             = comma
|             = OR
$             = End of the line
)             = End the 2nd group

编辑:添加了更多信息并简化了正则表达式。

有关详细信息,请参阅此帖子,并建议将其封装在函数中以便于重用:REGEX to select nth value from a list, allowing for nulls 这是我发现格式'[^,]+'有问题的帖子。不幸的是,它是最常见的正则表达式格式,是有关如何解析列表的问题的答案。我不禁想到'[^,]+'

返回的所有错误数据

答案 1 :(得分:13)

您可以使用regexp_substr()

select regexp_substr(val, '[^,]+', 1, 1) as val1, 
       regexp_substr(val, '[^,]+', 1, 2) as val2, 
       regexp_substr(val, '[^,]+', 1, 3) as val3, 
       . . .

我建议您在Excel(或其他电子表格)中生成255个数字的列,并使用电子表格生成SQL代码。

答案 2 :(得分:3)

如果您只有一行,并且有时间来创建

select * from (
  select rownum r , collection.*  
    from TABLE(cto_table(',','1.25, 3.87, 2, 19,, 1, 9, ')) collection
)
PIVOT ( 
  LISTAGG(column_value) within group (order by 1) as val 
  for r in (1, 2, 3, 4, 5, 6, 7, 8, 9, 10)
)

仅供参考:以下是如何创建cto_table功能:

CREATE OR REPLACE TYPE t_my_list AS TABLE OF VARCHAR2(100);
CREATE OR REPLACE
FUNCTION cto_table(p_sep in Varchar2, p_list IN VARCHAR2)
  RETURN t_my_list
AS
  l_string VARCHAR2(32767) := p_list || p_sep;
  l_sep_index PLS_INTEGER;
  l_index PLS_INTEGER := 1;
  l_tab t_my_list     := t_my_list();
BEGIN
  LOOP
    l_sep_index := INSTR(l_string, p_sep, l_index);
    EXIT
  WHEN l_sep_index = 0;
    l_tab.EXTEND;
    l_tab(l_tab.COUNT) := TRIM(SUBSTR(l_string,l_index,l_sep_index - l_index));
    l_index            := l_sep_index + 1;
  END LOOP;
  RETURN l_tab;
END cto_table;
/

答案 3 :(得分:1)

可以使用

分层查询。可以使用case和group by来完成数据透视。

$ docker-machine env