DBMS_LOB.LoadFromFile Проверка работоспособности - PullRequest
0 голосов
/ 11 июня 2018

Я думаю, что, может быть, я начинаю понимать процесс здесь, но это так странно, что я сомневаюсь в себе.Далее см. Комментарии о том, в чем суть путаницы.

DECLARE
    srcFile   BFILE := BFILENAME('SOME_DIR', 'xyz.pdf');
    fLen      NUMBER;
    dest      BLOB;

  BEGIN
    INSERT INTO SomeTable
    VALUES ( 1, 'xyz.pdf',  EMPTY_BLOB(), null ) -- Does this establish an active ref.?
    RETURNING pdf_file INTO dest;

    DBMS_LOB.OPEN(srcFile, dbms_lob.file_readonly);
    fLen := dbms_lob.getlength(srcFile);

    DBMS_LOB.LOADFROMFILE(dest, srcFile, fLen); -- Does this reach into the table row,
                                                -- and fill it with the file’s contents?
    dbms_lob.close(srcFile);
    commit;
  END;

И есть ли следующий способ обновления BLOB в строке, которая уже существует?

DECLARE
    srcFile   BFILE := BFILENAME('SOME_DIR', 'xyz.pdf');
    fLen      NUMBER;
    dest      BLOB;

  BEGIN
    SELECT pdf_file INTO dest -- Does this est. an active reference?
    FROM SomeTable   
    WHERE ID = 1; -- ( <———<<< ' = 1' is just for example.)

    DBMS_LOB.OPEN(srcFile, dbms_lob.file_readonly);
    fLen := dbms_lob.getlength(srcFile);

    DBMS_LOB.LoadFromFile(dest, srcFile, fLen); -- Does this reach into the row,
                                                -- filling it w/ the file’s contents?
    dbms_lob.close(srcFile);
    commit;
  END;

Похоже, это больше похоже на обработку .NET адаптеров базы данных и функций FileInfo, DirInfo.Но я не видел такой философии в Oracle где-либо еще.

Я бы ожидал

  BLOBVariable = LoadFromFile(srcLocator, byteCnt); -- where the func. rtrns a val,

, следующий за

  INSERT INTO SomeTable (pdf_file)
  VALUES              (BLOBVariable);

Я вижуэто право?Если да, то когда ссылка отменяется?Кажется, я помню, что читал, что 'commit' не нужен при использовании 'LoadFromFile?'

И сегодня утром я вижу пример (без объяснения причин), который выбирает поле BLOB в переменную, используя FOR UPDATEзаблокировать запись.Может быть, хорошо понять, что с этим ...

    -- Lock the record
    execute immediate 'SELECT pdf_file INTO dest
                       FROM   image_blobs
                       WHERE  itl_image_blob_id = :1
                       FOR UPDATE' 
    INTO v_blob_data
    using < the row identifier goes here >;

    -- Read the file
    dbms_lob.loadfromfile(dests, srcFile, fLen);

    -- update the blob field
    execute immediate '
        UPDATE image_blobs
        SET    pdf_file = :1
        WHERE  itl_image_blob_id = :2' 
    using dest, < row identifier >;

1 Ответ

0 голосов
/ 18 июня 2018

Следующее показало мне, что мое здравомыслие не повреждено, и что оператор INSERT действительно создает ссылку, которую использует loadfromfile().Этот ответ относительно оператора INSERT достаточно хорош для моих целей, поэтому я собираюсь предположить, что функциональность UPDATE аналогична.'IMPORT_DIR' является именованным каталогом базы данных в базе данных.

DROP TABLE TEST_BLOBS purge; -- 'purge' prevents loading up the recycle bin.
CREATE TABLE TEST_BLOBS (
  ID            VARCHAR(4),
  F_NAME        VARCHAR2(50 BYTE),
  CONTENT_TYPE  VARCHAR2(50 BYTE),
  F_SIZE        NUMBER(10),
  BLOB_DATA     BLOB DEFAULT empty_blob() );

DECLARE
    dest    blob;                 tmp_id  varchar2(4);
    v_f_sz  NUMBER(10);           b_file  BFILE := NULL;
    fName   VARCHAR2 (50);

    CURSOR get_items IS SELECT filename FROM My_Table_of_names;
BEGIN
  FOR a_rec IN get_items LOOP
    fName  := a_rec.filename;
    b_file := BFILENAME ( 'IMPORT_DIR', fName );
    IF DBMS_LOB.fileexists (b_file) = 0 THEN  
      -- Report the problem and exit.
    END IF;

    DBMS_LOB.fileopen ( b_file, DBMS_LOB.file_readonly );
    v_f_sz := DBMS_LOB.getlength (b_file);

    INSERT INTO test_blobs (            ID,   content_type,   f_name, f_size,   blob_data      )
    VALUES                 ( <Some_ID_val>, 'application/pdf',  fName,  v_f_sz, empty_blob()   )
    RETURNING blob_data, ID   INTO dest, tmp_id;

    -- Table field contains data, (HUGEBLOB),
    -- when debugging and loop ends here.

    DBMS_LOB.loadfromfile(dest, b_file, v_f_sz);
    DBMS_LOB.close(b_file);

    -- After running this, the variable tmp_id has the value that was assigned in the INSERT statement,
    -- and the table’s fields, 'BLOB_Data', have the contents of the files.   dest is indeed a reference
    -- to the row and field, allowing LoadFromFile() to put data into the table.
  END LOOP;
END;
/
...