Oracle 拆分字符串 + 字符串指定字符个数统计

目录

    • regexp_substr函数
    • 统计字符串指定字符个数
    • 模拟split拆分检索

regexp_substr函数

regexp_substr(srcstr, pattern, position, occurrence, modifier)

-srcstr:需要处理的字符串;
-pattern:正则表达式;
-position:起始位置,从第几个字符开始正则表达式匹配。默认为1;
-occurrence:标识第几个匹配组,默认为1;
-modifier:'i’不区分大小写检索;‘c’区分大小写检索。默认为’c’ 。

occurrence设为level,会得到分割后的所有结果。level是oracle的关键字,表示查询深度,用来实现层级查询。

统计字符串指定字符个数

length(v_numlist) - length(replace(v_numlist, ',', ''))

v_numlist是原字符串,求出逗号的个数
通过求出原字符串长度和删除指定字符的字符串长度,相减获得统计个数

模拟split拆分检索

假设表tab1中仅有一行记录,内容如下:

Numlist Num Posi
23,56,90,543,91 91

其中Posi列的值为空
求Num列的数字在Numlist列中第一次出现时是第几个数字,并把结果更新到Posi列中。如果找不到,则写-1。
例如,当Numlist = ‘23,56,90,543,91’,Num = 91时,Posi = 5;
当Numlist = ‘23,56,90,543,91’,Num = 54时,Posi = -1。
注意:尽管543中有54这个数字,但并不认为54出现过。

根据这些条件,使用字符串拆分或许是最快的办法。

drop table tab1;
create table tab1(
    numlist varchar2(30),
    num varchar2(30),
    posi number
);
insert into tab1 values('23,56,90,543,91', '91', null);

set serveroutput on
declare
    v_numlist tab1.numlist % type;
    v_num tab1.num % type;
    v_posi number;
begin
    select numlist, num
    into v_numlist, v_num
    from tab1;
    
    select index_level
    into v_posi
    from (
        select regexp_substr(v_numlist, '[^,$]+', 1, level) str, level index_level
        from dual
        connect by level <= length(v_numlist) - length(replace(v_numlist, ',', '')) + 1
    )
    where str = v_num and rownum <= 1;
    update tab1 set posi = v_posi where num = v_num;
    exception
        when no_data_found then 
            update tab1 set posi = -1 where num = v_num;
end;
/
select * from tab1;

注意一点,统计是统计指定字符个数,拆分后的字符串个数是统计值+1
所以,level <= length(v_numlist) - length(replace(v_numlist, ‘,’, ‘’)) + 1

你可能感兴趣的:(Oracle,SQL,正则表达式,Oracle,SQL,正则表达式,字符串拆分,字符串统计)