Search code examples
oracle-databaseplsqlplsqldeveloper

Reading clob line by line with pl\sql


In my project i use oracle as primary database and i've faced a problem with parsing clob. So suppose we have a clob with value

   aaaaaa
   cccccc
   bbbbbb

And it's stored in table test ...

I need to write plsql procedure to get this clob and split it so that i will have array with three items [aaaaaa,cccccccc,bbbbbbb].

Is there any possible solutions?


Solution

  • Here is a piece of code that works. I suggest that you use explicit cursors instead of implicit ones (FOR i IN (select...)), for performance purpose.

    First here is the script to create testcase.

    create table test (c clob);
    
    insert into test (c) values (
    'azertyuiop
    qsdfghjklm
    wxcvbn
    ');
    

    Then here is the script to read line by line Clob :

    /* Formatted on 28/08/2012 14:16:52 (QP5 v5.115.810.9015) */
    declare
        nStartIndex number := 1;
        nEndIndex number := 1;
        nLineIndex number := 0;
        vLine varchar2(2000);
    
        cursor c_clob is
        select c from test;
    
        c clob;
        -------------------------------
        procedure printout
           (p_clob in out nocopy clob) is
          offset number := 1;
          amount number := 32767;
          len    number := dbms_lob.getlength(p_clob);
          lc_buffer varchar2(32767);
          i pls_integer := 1;
        begin
          if ( dbms_lob.isopen(p_clob) != 1 ) then
            dbms_lob.open(p_clob, 0);
          end if;
          amount := instr(p_clob, chr(10), offset);
          while ( offset < len )
          loop
            dbms_lob.read(p_clob, amount, offset, lc_buffer);
            dbms_output.put_line('Line #'||i||':'||lc_buffer);
           offset := offset + amount;
           i := i + 1;
          end loop; 
              if ( dbms_lob.isopen(p_clob) = 1 ) then
            dbms_lob.close(p_clob);
          end if; 
        exception
          when others then
             dbms_output.put_line('Error : '||sqlerrm);
        end printout;
        ---------------------------
    begin
        dbms_output.put_line('-----------');
        open c_clob;
        loop
           fetch c_clob into c;
           exit when c_clob%notfound;
           printout(c);
        end loop;
        close c_clob;
    end;
    

    'amount' variable is used to detect end of line position. Be carfull, in some case the end of line is CHR(10)||CHR(13) (CR + LF), and in some other cases it is only CHR(10).