Joh*_*Doe 2 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);
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;
Run Code Online (Sandbox Code Playgroud)
但是当我执行该过程时,出现以下错误:
Error:ORA-22285: non-existent directory or file FILEOPEN operation
Run Code Online (Sandbox Code Playgroud)
C:\XMLS\如果我的 XML 文件存在,我不明白为什么路径确实存在。
我得到了带有查询结果的路线。
查询的这个结果被分配给一个名为 的变量GRUPO,这是我在过程开始时声明的,然后我将此变量作为参数放在函数中BFILENAME,该函数最初包含directoryy filename。您可以在此处查看文档BFILENAME
l_loc := BFILENAME(GRUPO,'ES0000251446027471.xml');
Run Code Online (Sandbox Code Playgroud)
除此之外,我在 Oracle 中创建了一个目录,如下所示:
CREATE OR REPLACE DIRECTORY DIR_XML as 'C:\XMLS\';
Run Code Online (Sandbox Code Playgroud)
我也给了目录权限
GRANT ALL ON DIRECTORY DIR_XML TO PUBLIC;
Run Code Online (Sandbox Code Playgroud)
我已经在这个问题上浪费了很多时间,我找不到任何解决方案。任何帮助将不胜感激。
错误:ORA-22285:文件打开操作的目录或文件不存在
如果我的 XML 文件存在,我不明白为什么路径 C:\XMLS\ 确实存在。
原因是 XML 文件存在于您的本地硬盘驱动器上,但 PL/SQL 运行在数据库服务器上。如果您想通过DBMS_LOB.
从您链接到的文档中:
- “目录”是一个数据库对象,用作文件实际所在的服务器文件系统上完整路径名的别名。
所以GRUPO应该评估为'DIR_XML'而不是'C:\XMLS'. 该文档的示例中显示了这一点。(有一种基于存储为数据库参数的路径的旧目录前对象机制,但安全性较低......)
如果您只有路径,则可以查找目录名称:
select directory_name from all_directories where directory_path = 'C:\XMLS'
Run Code Online (Sandbox Code Playgroud)
请记住,目录路径不必是唯一的,因此您可能需要处理重复项。
但正如@Matthew 已经解释的那样,正如文档所说(强调):
目录对象指定服务器文件系统上目录的别名,其中...
数据库只能查看其自身文件系统(本地或共享)上的文件,而不能查看客户端文件系统上的文件。如果您也在本地运行数据库,则没有区别(尽管目录和文件权限仍然很重要)。如果您正在访问远程数据库,那么它看不到您的客户端 C: 驱动器,并且如果您提供目录对象名称,您仍然会得到类似的信息:
ORA-22288: file or LOB operation FILEOPEN failed
No such file or directory
Run Code Online (Sandbox Code Playgroud)
您必须将 XML 文件放在操作系统帐户可以访问的数据库服务器上的目录中,并创建一个指向服务器上该位置的目录对象;然后引用目录对象名称,而不是底层文件系统路径。