如何解决Oracle DBMS_LOB的问题

如何解决Oracle DBMS_LOB的问题,oracle,plsql,procedure,Oracle,Plsql,Procedure,我试图将XML文件的信息保存在数据库表中,并使用以下步骤: create or replace PROCEDURE P_FILEUPLOAD_XML (P_CMTT_CODE IN NUMBER DEFAULT 15, P_TEXT IN VARCHAR2, P_TEXT_NAR IN VARCHAR2, P_PATH IN VARCHAR2, P_FILENAME IN VARCHAR2, P_RET_VAL OUT NUMBER) IS GRUPO VARCHAR2(20);

我试图将XML文件的信息保存在数据库表中,并使用以下步骤:

create or replace PROCEDURE P_FILEUPLOAD_XML (P_CMTT_CODE IN NUMBER DEFAULT 15, P_TEXT IN VARCHAR2, P_TEXT_NAR IN VARCHAR2, P_PATH IN VARCHAR2, P_FILENAME IN VARCHAR2, P_RET_VAL OUT NUMBER) IS

GRUPO       VARCHAR2(20);
l_dir       CONSTANT VARCHAR2(35) := P_PATH;
l_fil       CONSTANT VARCHAR2(30) := P_FILENAME; 
l_loc       BFILE; -- Pointer to the BFILE
l_ret       BOOLEAN := FALSE; -- Return value
l_pos       NUMBER := 1; -- Current position in the file (file begins at position 1)
l_sum       number default 8000; -- Amount of characters have been read
l_buf       VARCHAR2(32767); -- Read Buffer
l_sen       CONSTANT RAW(100) := UTL_RAW.CAST_TO_RAW(CHR(10)); -- Character at the end of the file is NEWLINE (ascii = 10)
l_end       NUMBER; -- End of the current word which will be read
l_counter NUMBER := 0; -- Counter for line sequence
l_line      VARCHAR2(32767); -- Contains the info line by line for insert

BEGIN

SELECT TEXTO INTO GRUPO FROM gzvcatg
         UNPIVOT ((codigo, texto) FOR gzvcatg_external_code IN (
            (gzvcatg_external_code1, gzvcatg_desc1) AS '1')
         WHERE GZVCATG_GROUP = 'TIT_ELEC'
            AND CODIGO = 'PATH';

     -- Mapping the physical file with the pointer to the BFILE
        l_loc := BFILENAME(GRUPO,'ES0000251446027471.xml');

    -- Open the file in READ_ONLY mode
       DBMS_LOB.OPEN(l_loc,DBMS_LOB.LOB_READONLY);
       LOOP
          l_counter := l_counter + 1; -- Counter for sequence
          -- Calculate the end of the current word
          l_end := DBMS_LOB.INSTR(l_loc,l_sen,l_pos,1);

          -- Process end-of-file
          IF (l_end = 0) THEN
            l_end := DBMS_LOB.INSTR(l_loc,l_sen,l_pos,1);
            l_sum := l_end - l_pos - 1;
            EXIT;
          END IF;

          -- Read until end-of-file
          l_sum := l_end - l_pos;
          DBMS_LOB.READ(l_loc,l_sum,l_pos,l_buf);
          l_line := UTL_RAW.CAST_TO_VARCHAR2(l_buf);

        BEGIN 
            INSERT INTO SPRCMNT (
                         SPRCMNT_CMTT_CODE,
                         SPRCMNT_TEXT,
                         SPRCMNT_TEXT_NAR)
                                VALUES(P_CMTT_CODE,
                                       P_TEXT,
                                       P_TEXT_NAR);    
             EXCEPTION WHEN OTHERS THEN 
             ROLLBACK;           
        END;
    END LOOP;    
EXCEPTION
    WHEN OTHERS THEN
        dbms_output.put_line('Error:' || SQLERRM);
        DBMS_LOB.CLOSE(l_loc);
        P_RET_VAL := 3;
        dbms_output.put_line('P_RET_VAL:' || P_RET_VAL);    
END;
但是,当我执行该过程时,会出现以下错误:

Error:ORA-22285: non-existent directory or file FILEOPEN operation
如果我的XML文件存在,我不理解为什么路径
C:\XMLS\
确实存在

我得到带有查询结果的路线

这个查询结果被分配给一个名为
GRUPO
的变量,这是我在过程开始时声明的,然后我将此变量作为参数放入函数
BFILENAME
,函数最初包含
目录
y
文件名
。您可以查看文档
BFILENAME

除此之外,我还在Oracle中创建了一个目录,如下所示:

CREATE OR REPLACE DIRECTORY DIR_XML as 'C:\XMLS\';
我还授予了该目录的权限

GRANT ALL ON DIRECTORY DIR_XML TO PUBLIC;
我在这个问题上浪费了很多时间,找不到任何解决办法。任何帮助都将不胜感激

错误:ORA-22285:文件打开操作的目录或文件不存在

如果我的XML文件存在,我不理解为什么路径C:\XMLS\存在

原因是XML文件存在于本地硬盘上,但PL/SQL在数据库服务器上运行。如果您想通过
DBMS\u LOB

从以下位置访问该文件,则需要将该文件放在数据库服务器上:

  • “目录”是一个数据库对象,用作文件实际所在服务器文件系统上完整路径名的别名
因此,
GRUPO
应该计算为
'DIR\uxml'
,而不是
'C:\XMLS'
。这在该文档的示例中显示。(有一个旧的预目录对象机制,它基于存储为数据库参数的路径,但不太安全…)

如果只有路径,则可以查找目录名:

select directory_name from all_directories where directory_path = 'C:\XMLS'
请记住,目录路径不必是唯一的,因此您可能需要处理重复的路径

但正如@Matthew已经解释过的,以及(重点补充):

目录对象为服务器文件系统上的目录指定别名,其中

数据库只能看到自己的文件系统上的文件(本地或共享),而不能看到客户端文件系统上的文件。如果您也在本地运行DB,则没有区别(尽管目录和文件权限仍然很重要)。如果您正在访问一个远程数据库,那么它就看不到您的客户机C:drive,并且如果您提供目录对象名称,您仍然会得到如下结果:

ORA-22288: file or LOB operation FILEOPEN failed
No such file or directory
您必须将XML文件放在操作系统帐户有权访问的DB服务器上的目录中,并创建指向服务器上该位置的目录对象;然后引用目录对象名,而不是底层文件系统路径。

请尝试以下操作:

  • 将DIR_XML文件夹设为共享文件夹,对所有人都具有读/写权限
  • 使用目录路径“\\192.168.2.100\dir\u xml”而不是“C:\dir\u xml”

  • 将192.168.2.100替换为存在C:\DIR\u XML的IP地址或计算机名

    不要使用C:\path。使用网络路径。确保可以从sql客户端计算机以及Oracle server访问\123.3.3.3\xml\文件夹。如果检查Ora tnsnames.Ora文件,可以找到数据库服务器名称。 检查服务器是否创建共享文件夹,然后使用它。 192.168.1.0\xml\ Tnsnames.ora示例

    ORA11 =
     (DESCRIPTION = 
       (ADDRESS_LIST =
         (ADDRESS = (PROTOCOL = TCP)(HOST = 192.168.1.0)(PORT = 1521))
       )
     (CONNECT_DATA =
       (SERVICE_NAME = ORA12)
     )
    )
    

    数据库服务器上存在目录“C:\XMLS\”,对吗?本地硬盘上的“C:\XMLS\”目录无法工作。在数据库服务上运行PL/SQL块时,该目录存在于我的本地硬盘上,文件访问将查看数据库服务器上的驱动器。这就是你的问题。你知道如何获得
    l_loc
    变量带来的结果,我如何使用
    dbms_输出
    ,来验证路由是否正确,它给我带来了比我更好、更完整的答案。
    ORA11 =
     (DESCRIPTION = 
       (ADDRESS_LIST =
         (ADDRESS = (PROTOCOL = TCP)(HOST = 192.168.1.0)(PORT = 1521))
       )
     (CONNECT_DATA =
       (SERVICE_NAME = ORA12)
     )
    )