如何解决Oracle DBMS_LOB的问题
我试图将XML文件的信息保存在数据库表中,并使用以下步骤:如何解决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);
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服务器上的目录中,并创建指向服务器上该位置的目录对象;然后引用目录对象名,而不是底层文件系统路径。请尝试以下操作:
将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)
)
)