返回值时使用255个逗号分隔的值。是否有一种不需要255个substr就可以将这些列分割成列的简单方法?
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 ...发布于 2015-07-16 20:59:39
您可以使用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代码。
发布于 2015-07-16 21:17:55
当心!如果列表中有一个空元素,并且您想要该项或它后面的一个,'[^,]+'格式的'[^,]+'表达式将不会返回预期值。考虑这个示例,其中第4个元素为NULL,我希望第5个元素,因此期望返回'5‘:
SQL> select regexp_substr('1,2,3,,5,6', '[^,]+', 1, 5) from dual;
R
-
6惊喜吧!它返回第5个非空元素,而不是实际的第5个元素!返回的数据不正确,甚至可能抓不到它。试一试:
SQL> select regexp_substr('1,2,3,,5,6', '(.*?)(,|$)', 1, 5, NULL, 1) from dual;
R
-
5因此,上面更正的REGEXP_SUBSTR表示要查找0或更多逗号分隔字符的第5次出现,后面跟着逗号或行尾(允许下一个分隔符,无论是逗号还是行尾),并在被发现时返回第一个子组(不包括逗号或行尾的数据)。
搜索匹配模式'(.*?)(,|$)'解释道:
( = 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,这是我发现'[^,]+'格式有问题的帖子。不幸的是,这是您最常看到的regex格式,用于回答有关如何解析列表的问题。想到'[^,]+'返回的所有不正确的数据,我会不寒而栗!
发布于 2017-04-12 07:43:39
如果您只有一行,并且有时间创建
cto_table function to split a string on any separator,然后可以使用PIVOT + LISTAGG进行如下操作: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)
)FYI:下面是如何创建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;
/https://stackoverflow.com/questions/31464275
复制相似问题