0

我有一个名为 TVL_DETAIL 的表,其中包含 TVL_CD_LIST 列。列 TVL_CD_LIST 包含三个记录:
TVL_CD_LIST:
M1180_Z6827
K5900_Z6828
I2510

我使用以下代码尝试仅返回值(因此不包括下划线):

SELECT
TVL_CD_LIST   
FROM TVL_DETAIL  
WHERE TVL_CD_LIST IN (SELECT regexp_substr(TVL_CD_LIST,'[^_]+', 1, level) FROM DUAL
CONNECT BY regexp_substr(TVL_CD_LIST,'[^_]+', 1, level) IS NOT NULL)

我期望在单独的行中看到返回的是:
M1180
Z6827
K5900
Z6828
I2510

但它只返回 I2510(这是不包含下划线的原始值)。

我究竟做错了什么?任何帮助表示赞赏。谢谢!

4

2 回答 2

0

要回答您的问题,您正在查询与子元素匹配的列表,并且只会在列表由一个元素组成的情况下发生。您真正想要选择的是子元素本身。

注意:这里解释为什么使用正则表达式解析字符串'[^_]+'不好:https ://stackoverflow.com/a/31464699/2543416

您要解析列表,选择元素:

SQL> with TVL_DETAIL(TVL_CD_LIST) as (
     select 'M1180_Z6827' from dual union
     select 'K5900_Z6828' from dual union
     select 'I2510' from dual
   )
   SELECT distinct regexp_substr(TVL_CD_LIST, '(.*?)(_|$)', 1, level, NULL, 1) element
   FROM TVL_DETAIL
   CONNECT BY level <= LENGTH(regexp_replace(TVL_CD_LIST, '[^_]', '')) + 1;
   -- 11g  CONNECT BY level <= regexp_count(TVL_CD_LIST, '_') + 1;

ELEMENT
-----------
Z6827
K5900
M1180
I2510
Z6828

SQL>

如果您想按行和行内的元素进行跟踪,这很酷:

SQL> with TVL_DETAIL(row_nbr, TVL_CD_LIST) as (
     select 1, 'M1180_Z6827' from dual union
     select 2, 'K5900_Z6828' from dual union
     select 3, 'I2510' from dual
   )
   SELECT row_nbr, column_value substring_nbr,
          regexp_substr(TVL_CD_LIST, '(.*?)(_|$)', 1, column_value, NULL, 1) element
   FROM TVL_DETAIL,
     TABLE(
       CAST(
         MULTISET(SELECT LEVEL
                  FROM dual
                  CONNECT BY level <= LENGTH(regexp_replace(TVL_CD_LIST, '[^_]', '')) + 1
                  -- 11g CONNECT BY LEVEL <= REGEXP_COUNT(TVL_CD_LIST, '_')+1
                  ) AS sys.OdciNumberList
           )
     )
   order by row_nbr, substring_nbr;

   ROW_NBR SUBSTRING_NBR ELEMENT
---------- ------------- -----------
         1             1 M1180
         1             2 Z6827
         2             1 K5900
         2             2 Z6828
         3             1 I2510

SQL>

编辑:糟糕,编辑为使用 10g,因为 REGEXP_COUNT 直到 11g 才可用。

于 2016-12-07T21:55:35.043 回答
0

您使用的查询创建了列表,但您正在使用子句将记录列表与其自身的列进行比较in,例如M1180orZ6827不能等于M1180_Z6827等 for K5900_Z6828I2510只有一个值,所以它被匹配。

如果您的要求正是您在所需输出中提到的,您可以使用以下查询。

SQL> WITH tvl_detail AS
      2   (SELECT 'M1180_Z6827' tvl_cd_list FROM dual
      3    UNION ALL
      4    SELECT 'K5900_Z6828' FROM dual
      5    UNION ALL
      6    SELECT 'I2510' FROM dual)
      7   ---------------------------
      8   --- End of data preparation
      9   ---------------------------
     10  SELECT regexp_substr(tvl_cd_list, '[^_]+', 1, LEVEL) AS tvl_cd_list
     11    FROM tvl_detail
     12  CONNECT BY regexp_substr(tvl_cd_list, '[^_]+', 1, LEVEL) IS NOT NULL
     13         AND PRIOR tvl_cd_list = tvl_cd_list
     14         AND PRIOR sys_guid() IS NOT NULL;

输出:

TVL_CD_LIST
--------------------------------------------
I2510
K5900
Z6828
M1180
Z6827
于 2016-12-08T09:06:10.420 回答