데이터 베이스 MySQL 튜토리얼 oracel中字符串分割成集合详解

oracel中字符串分割成集合详解

Jun 07, 2016 pm 05:49 PM
문자열 분할

文章分享一篇关于自己的学习笔记,oracel中字符串分割成集合详解有需要学习的同学可以看看。

首先分别使用两种方式构造两个函数

 代码如下 复制代码

-- use conventional plsql
create or replace function f_str2list_pls
(
  p_str       varchar2,
  p_separator varchar2 default ','
) return my_tk_str_tab_type is
  l_idx  pls_integer := 0;
  l_str  varchar2(32767) := trim(p_str);
  l_elmt varchar2(100) := null;
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
begin
  while l_str is not null loop
    l_idx := instr(l_str, p_separator);
    if l_idx = 0 then
      l_elmt := l_str;
      l_str  := null;
    else
      l_elmt := substr(l_str, 1, l_idx - 1);
      l_str  := substr(l_str, l_idx + 1);
    end if;
 
    l_list.extend;
    l_list(l_list.last) := trim(l_elmt);
  end loop;

  return l_list;
end;
/

-- use single sql
create or replace function f_str2list_sql
(
  p_str       varchar2,
  p_separator varchar2 default ','
) return my_tk_str_tab_type is
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
begin
  substr(a.str,
                instr(p_separator || a.str, p_separator, 1, rn),
                instr(a.str || p_separator, p_separator, 1, rn) -
                instr(p_separator || a.str, p_separator, 1, rn)) q
    bulk collect into l_list
    from (select p_str as str from dual) a,
         (select rownum rn from dual connect by rownum    where instr(p_separator || a.str, p_separator, 1, rn) > 0;

  return l_list;
end;
/

 

 


确认两种方法完成同样的功能

 

----------------------------------------------------

 代码如下 复制代码

-- same result
declare
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
  l_str varchar2(1000) := 'a,b,c';
begin
  l_list := f_str2list_pls(l_str,',');
  for i in 1..l_list.count loop
    dbms_output.put_line(l_list(i));
  end loop;
 
  dbms_output.put_line('');
  l_list := f_str2list_sql(l_str,',');
  for i in 1..l_list.count loop
    dbms_output.put_line(l_list(i));
  end loop;
end;
/

SQL> set serveroutput on
a
b
c
 
a
b
c

 

 

 

 

 

我们知道在PL/SQL和SQL里面,varchar2类型的长度限制是不同的。那么这两种方法是否也存在同样的限制?验证一下

先测试PL/SQL版本

 

-- 这里使用单个字母作为元素,加上必要的逗号分割符,一个元素占用长度2. PL/SQL中上限32767。当取(32767/2 = 16383)个元素的时候,运行成功
----------------------------------------------------

 代码如下 复制代码

-- pls versions tring length limit
declare
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
  l_str  varchar2(32767) := '';
  l_max  pls_integer := 16383;
begin
  -- construct string
  for i in 1 .. l_max loop
    l_str := l_str || ',' || 'a';
  end loop;
  l_str := substr(l_str, 2);

   l_list := f_str2list_pls(l_str, ',');
end;
/

PL/SQL procedure successfully completed

-- 增加一个元素,当取(16384)个元素的时候,超过了varchar2的限制,所以无法运行。(构造字符串就报错了,这个时候其实还没有调用f_str2list_pls)
 代码如下 复制代码

declare
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
  l_str  varchar2(32767) := '';
  l_max  pls_integer := 16384;
begin
  -- construct string
  for i in 1 .. l_max loop
    l_str := l_str || ',' || 'a';
  end loop;

  l_list := f_str2list_pls(l_str, ',');
end;
/

ORA-06502: PL/SQL: numeric or value error: character string buffer too small
ORA-06512: at line 9

-- 修改一下代码,实际调用这个函数。从结果可以看到,f_str2list_pls()内部报超长错误
declare
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
  l_str  varchar2(32767) := '';
  l_max  pls_integer := 16383;
begin
  -- construct string
  for i in 1 .. l_max loop
    l_str := l_str || ',' || 'a';
  end loop;

  l_list := f_str2list_pls(l_str||',a', ',');
end;
 
ORA-06502: PL/SQL: numeric or value error: character string buffer too small
ORA-06512: at "T2.F_STR2LIST_PLS", line 7
ORA-06512: at line 12

 

 

 


下面测试SQL版本

 

 代码如下 复制代码

-- 因为SQL中varchar2上限是4000,所以使用2000个元素。一切正常。
----------------------------------------------------
-- sql versions tring length limit
declare
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
  l_str  varchar2(32767) := '';
  l_max  pls_integer := 2000;
begin
  -- construct string
  for i in 1 .. l_max loop
    l_str := l_str || ',' || 'a';
  end loop;
  l_str := substr(l_str, 2);

   l_list := f_str2list_sql(l_str, ',');
end;
/

PL/SQL procedure successfully completed

-- 直接增加到2001个元素,发现f_str2list_sql()内部报错,说明构造字符串的成功传入,但是函数不能处理
 代码如下 复制代码

declare
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
  l_str  varchar2(32767) := '';
  l_max  pls_integer := 2001;
begin
  -- construct string
  for i in 1 .. l_max loop
    l_str := l_str || ',' || 'a';
  end loop;
  l_str := substr(l_str, 2);

   l_list := f_str2list_sql(l_str, ',');
end;
 
ORA-01460: unimplemented or unreasonable conversion ed
ORA-06512: at "T2.F_STR2LIST_SQL", line 8
ORA-06512: at line 13

备注:ORA-01460的描述很不清楚,其实这里就是varchar2超长

 

 

 

 

 

测试比较两种方式在不同数据量下的性能

 

 代码如下 复制代码

----------------------------------------------------
-- performance test
declare
  l_list my_tk_str_tab_type := my_tk_str_tab_type();
  l_str  varchar2(32767) := '';
  l_max  pls_integer := 2000;
begin
  -- construct string
  for i in 1 .. l_max loop
    l_str := l_str || ',' || 'a';
  end loop;
  l_str := substr(l_str, 2);

  -- warm up before actually calculation
  l_list := f_str2list_sql(l_str, ',');
  l_list := f_str2list_pls(l_str, ',');
 
  -- begin calc and diff
  my_rs.rs_start;
  -- 1. pls version
  l_list := f_str2list_pls(l_str, ',');
  my_rs.rs_middle;
  -- 2. sql version
  l_list := f_str2list_sql(l_str, ',');
  my_rs.rs_stop();
end;
/

 

 

 


我们分别测试当元素个数为100, 200, 500, 1000, 2000的情况。每种情况进行3-5次然后取平均

元素个数

PL/SQL 运行时间 (1/100 second)

SQL 运行时间 (1/100 second)

PCT

100

1

1

100%

200

1

1 - 2

50% - 100%

500

1 - 2

2 - 3

30% - 50%

1000

1

4

25%

2000

2

10

20%

总结:

SQL版本的书写简便,并且可以脱离PL/SQL环境直接使用(单条SQL进行行列转换)。

SQL版本只能处理长度小于4000的字符串,实际最大只能包含2000个元素。PL/SQL版本可以处理长度小于32767的字符串。

小数据量情况下,两者性能相当。随着数据量增大,PL/SQL版本性能明显占优

 

 

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

인기 기사

<gum> : Bubble Gum Simulator Infinity- 로얄 키를 얻고 사용하는 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
Nordhold : Fusion System, 설명
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌
Mandragora : 마녀 트리의 속삭임 - Grappling Hook 잠금 해제 방법
3 몇 주 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

Python 3.x에서 Split() 함수를 사용하여 지정된 구분 기호에 따라 문자열을 분할하는 방법 Python 3.x에서 Split() 함수를 사용하여 지정된 구분 기호에 따라 문자열을 분할하는 방법 Jul 31, 2023 pm 08:33 PM

Python은 문자열을 처리하기 위한 많은 내장 함수를 제공하는 인기 있는 프로그래밍 언어입니다. 일반적으로 사용되는 함수 중 하나는 지정된 구분 기호에 따라 문자열을 여러 하위 문자열로 분할할 수 있는 Split() 함수입니다. 이 기사에서는 Python3.x에서 Split() 함수를 사용하는 방법을 소개합니다. Python에서 Split() 함수는 문자열 클래스의 내장 함수입니다. 기본 구문은 다음과 같습니다. string.split(separator,maxsplit)

폭발 함수를 사용하여 PHP에서 문자열을 분할하는 방법 폭발 함수를 사용하여 PHP에서 문자열을 분할하는 방법 Jun 26, 2023 pm 12:03 PM

PHP 언어에는 문자열을 빠르고 효율적으로 처리하는 데 도움이 되는 많은 기본 함수가 있습니다. 그중 폭발 기능은 매우 실용적인 문자열 분할 기능입니다. 지정된 구분 기호에 따라 문자열을 배열로 분할한 다음 보다 유연한 문자열 작업을 수행할 수 있습니다. 이 기사에서는 폭발 기능을 사용하여 PHP에서 문자열을 분할하는 방법을 소개합니다. 1. 분해 함수의 형식은 PHP 언어에서 분해 함수의 형식은 다음과 같습니다.

문자열의 각 문자가 하위 문자열에 표시되도록 하는 문자열의 최대 분할 길이 문자열의 각 문자가 하위 문자열에 표시되도록 하는 문자열의 최대 분할 길이 Aug 25, 2023 pm 02:41 PM

이 기사에서는 고유한 문자가 있는 문자열의 최대 분할 길이를 찾는 방법에 대한 문제를 살펴보겠습니다. 먼저 문제 설명을 이해한 다음 해당 알고리즘과 시간 복잡성을 포함하여 이 문제를 해결하기 위한 순진하고 효율적인 방법을 연구합니다. 마지막으로 C++로 솔루션을 구현하겠습니다. 문제 설명 주어진 문자열에서 문자열의 각 문자가 하나의 하위 문자열에만 나타나도록 문자열을 가능한 많은 하위 문자열로 분할합니다. 이러한 최대화 분할의 길이를 반환합니다. 순진한 접근 방식 순진한 접근 방식은 문자열을 반복하여 각 문자의 마지막 발생을 기록하는 것입니다. 그런 다음 문자열을 다시 반복하고 현재 문자의 마지막 항목이 발견되면 파티션을 만듭니다. 문자열을 저장하기 위해 배열을 초기화하는 알고리즘(순진함)

PHP에서 문자열을 배열로 나누는 방법 PHP에서 문자열을 배열로 나누는 방법 Jul 08, 2023 pm 01:49 PM

PHP에서 문자열을 배열로 분할하는 방법 PHP에서는 문자열을 처리하고 여러 부분으로 분할해야 하는 경우가 많습니다. 문자열을 배열로 분할하는 것은 문자열의 다양한 부분을 더 잘 처리하는 데 도움이 되는 일반적인 작업입니다. 이 기사에서는 PHP의 함수를 사용하여 문자열을 배열로 분할하는 방법을 배우고 몇 가지 코드 예제를 제공합니다. 폭발 함수를 사용하여 문자열을 배열로 분할합니다. PHP는 지정된 구분 기호에 따라 문자열을 분할할 수 있는 폭발이라는 함수를 제공합니다.

문자열을 짝수로 시작하고 최소 길이가 M인 K개의 하위 문자열로 분할하는 방법의 수를 계산합니다. 문자열을 짝수로 시작하고 최소 길이가 M인 K개의 하위 문자열로 분할하는 방법의 수를 계산합니다. Sep 09, 2023 pm 02:01 PM

이 문제에서는 주어진 문자열을 문제 설명에 주어진 조건을 만족하도록 K개의 하위 문자열로 나누는 방법을 계산합니다. 이 문제를 해결하기 위해 재귀를 사용하겠습니다. 또한 이 문제를 효율적으로 해결하기 위해 표 형식의 동적 프로그래밍 방법을 사용할 것입니다. 문제 설명 - bin_Str이라는 특정 길이의 문자열이 있습니다. 문자열에는 '0'부터 '9'까지의 숫자만 포함됩니다. 다음 조건을 만족하도록 문자열을 K개의 하위 문자열로 분할하는 방법의 수를 계산해야 합니다. 하위 문자열은 2자 이상을 포함해야 합니다. 각 하위 문자열의 첫 번째 문자는 짝수여야 하고 마지막 문자는 홀수여야 합니다. 예예제 입력M=2,K=2;bin_str="255687&q

문자열을 세 개의 하위 문자열로 분할할 수 있는지 확인합니다. 여기서 하나의 하위 문자열은 다른 두 하위 문자열의 하위 문자열입니다. 문자열을 세 개의 하위 문자열로 분할할 수 있는지 확인합니다. 여기서 하나의 하위 문자열은 다른 두 하위 문자열의 하위 문자열입니다. Sep 22, 2023 am 11:53 AM

이 문제에서는 세 번째 부분 문자열이 처음 두 부분 문자열의 부분 문자열이 될 수 있도록 주어진 문자열을 분할해야 합니다. 해결책을 생각해 봅시다. 세 번째 문자열은 처음 두 문자열이 세 번째 문자열의 모든 문자를 포함하는 경우에만 처음 두 문자열의 하위 문자열이 될 수 있습니다. 따라서 주어진 문자열에서 빈도가 3보다 큰 문자를 하나 이상 찾아야 하며 해당 단일 문자의 세 번째 하위 문자열을 가져올 수 있습니다. 문제 설명 - N개의 소문자 알파벳 문자를 포함하는 문자열 str이 제공됩니다. 문자열을 세 개의 부분 문자열 a, b, c로 분할하여 부분 문자열 c가 a와 b의 부분 문자열이 되도록 할 수 있는지 확인해야 합니다. 3개의 하위 문자열을 찾을 수 있는지 여부에 따라 "yes" 또는 "no"를 인쇄합니다.

PHP의 폭발 함수에서 보고된 오류를 해결하는 방법 PHP의 폭발 함수에서 보고된 오류를 해결하는 방법 Mar 11, 2024 am 11:45 AM

PHP에서 분해 함수에 의해 보고된 오류를 해결하는 방법에는 특정 코드 예제가 필요합니다. PHP에서 분해 함수는 지정된 구분 기호에 따라 문자열을 배열로 분할하는 데 사용되는 함수입니다. 그러나 분해 기능을 사용할 때 오류가 발생하는 경우가 있는데, 주로 전달된 매개변수가 기능 요구 사항을 충족하지 않기 때문입니다. 아래에서는 가능한 문제와 해결 방법을 자세히 논의하고 구체적인 코드 예제를 제공합니다. 분해 기능 사용 시 잘못된 매개변수 개수로 인해 발생하는 오류

PHP 문자열 함수 예: 문자열 분할 PHP 문자열 함수 예: 문자열 분할 Jun 20, 2023 pm 01:58 PM

PHP에는 많은 문자열 함수가 있으며, 그 중 문자열 분할 함수가 매우 일반적으로 사용됩니다. 문자열 분할 함수는 지정된 구분 기호에 따라 문자열을 분할하고 배열을 반환할 수 있습니다. 아래에서는 일반적으로 사용되는 몇 가지 문자열 분할 기능을 소개합니다. 폭발 함수 폭발 함수는 지정된 구분 기호에 따라 문자열을 분할하고 배열을 반환할 수 있습니다. 구문은 다음과 같습니다:explod(string$separator,string$string

See all articles