Using Create directory & UTL_FILE in Oracle

本文介绍如何使用Oracle数据库的Createdirectory特性实现文件的读写操作,包括创建目录、授予用户读写权限、通过utl_file包读写文件的具体步骤。

摘自:http://www.eygle.com/archives/2005/04/using_create_di.html

 

Create directory让我们可以在Oracle数据库中灵活的对文件进行读写操作,极大的提高了Oracle的易用性和可扩展性。
其语法为:
CREATE [OR REPLACE] DIRECTORY directory AS 'pathname';

本案例具体创建如下:

create or replace directory exp_dir as '/tmp';

目录创建以后,就可以把读写权限授予特定用户,具体语法如下:
GRANT READ[,WRITE] ON DIRECTORY directory TO username;

例如:
grant read, write on directory exp_dir to eygle;

此时用户eygle就拥有了对该目录的读写权限。

让我们看一个简单的测试:
SQL> create or replace directory UTL_FILE_DIR as '/opt/oracle/utl_file';


Directory created.

SQL> declare
  2    fhandle utl_file.file_type;
  3  begin
  4    fhandle := utl_file.fopen('UTL_FILE_DIR', 'example.txt', 'w');
  5    utl_file.put_line(fhandle , 'eygle test write one');
  6    utl_file.put_line(fhandle , 'eygle test write two');
  7    utl_file.fclose(fhandle);
  8  end;
  9  /

PL/SQL procedure successfully completed.

SQL> !
[oracle@jumper 9.2.0]$ more /opt/oracle/utl_file/example.txt 
eygle test write one
eygle test write two
[oracle@jumper 9.2.0]$ 
 

类似的我们可以通过utl_file来读取文件:
SQL> declare
  2    fhandle   utl_file.file_type;
  3    fp_buffer varchar2(4000);
  4  begin
  5    fhandle := utl_file.fopen ('UTL_FILE_DIR','example.txt', 'R');
  6  
  7    utl_file.get_line (fhandle , fp_buffer );
  8    dbms_output.put_line(fp_buffer );
  9    utl_file.get_line (fhandle , fp_buffer );
 10    dbms_output.put_line(fp_buffer );
 11    utl_file.fclose(fhandle);
 12  end;
 13  /
eygle test write one
eygle test write two

PL/SQL procedure successfully completed.

可以查询dba_directories查看所有directory.
SQL> select * from dba_directories;

OWNER                          DIRECTORY_NAME                 DIRECTORY_PATH
------------------------------ ------------------------------ ------------------------------
SYS                            UTL_FILE_DIR                   /opt/oracle/utl_file
SYS                            BDUMP_DIR                      /opt/oracle/admin/conner/bdump
SYS                            EXP_DIR                        /opt/oracle/utl_file


可以使用drop directory删除这些路径.
SQL> drop directory exp_dir;

Directory dropped

SQL> select * from dba_directories;

OWNER                          DIRECTORY_NAME                 DIRECTORY_PATH
------------------------------ ------------------------------ ------------------------------
SYS                            UTL_FILE_DIR                   /opt/oracle/utl_file
SYS                            BDUMP_DIR                      /opt/oracle/admin/conner/bdump
### Oracle存储过程导出BLOB数据的方法 在Oracle数据库中,可以通过编写PL/SQL存储过程实现对BLOB数据的读取和导出操作。以下是具体方法以及示例代码。 #### 方法概述 为了从Oracle数据库中提取BLOB数据并将其保存到文件系统或其他位置,可以利用`UTL_FILE`包将二进制数据写入操作系统上的文件[^1]。此过程中需要注意权限设置,因为`UTL_FILE`需要访问目标目录路径对应的DBA创建的DIRECTORY对象。 #### 示例代码 以下是一个完整的存储过程示例,用于将指定表中的BLOB字段导出至服务器端的一个文件: ```plsql CREATE OR REPLACE PROCEDURE export_blob_to_file ( p_table_name IN VARCHAR2, p_column_name IN VARCHAR2, p_id_value IN NUMBER, -- 假设有一个唯一ID列作为条件筛选某条记录 p_directory IN VARCHAR2, p_filename IN VARCHAR2 ) AS l_bfile BFILE; l_blob BLOB; l_dest_offset INTEGER := 1; -- 文件起始偏移量 l_src_offset INTEGER := 1; -- Blob起始偏移量 l_lang_context INTEGER := DBMS_LOB.DEFAULT_LANG_CTX; l_warning INTEGER; BEGIN -- 查询获取Blob数据 EXECUTE IMMEDIATE 'SELECT ' || p_column_name || ' FROM ' || p_table_name || ' WHERE id = :id' INTO l_blob USING p_id_value; IF DBMS_LOB.GETLENGTH(l_blob) > 0 THEN -- 打开目标文件准备写入 l_bfile := UTL_FILE.FOPEN(p_directory, p_filename, 'wb', 32767); -- 将Blob内容复制到文件 DBMS_LOB.CREATETEMPORARY(l_blob, TRUE); DBMS_LOB.FILECOPY( DEST_OFFSET => l_dest_offset, SRC_BLOB => l_blob, AMOUNT => DBMS_LOB.GETLENGTH(l_blob), FILE_NAME => l_bfile ); -- 关闭文件句柄 UTL_FILE.FCLOSE(l_bfile); END IF; EXCEPTION WHEN OTHERS THEN IF UTL_FILE.IS_OPEN(l_bfile) THEN UTL_FILE.FCLOSE(l_bfile); END IF; RAISE; END; / ``` 以上代码定义了一个通用化的存储过程 `export_blob_to_file` ,它接受五个参数:表名、BLOB字段名称、主键值(假设存在一个名为`id` 的主键)、目标目录以及输出文件名 。该程序会先查询符合条件的一行数据取出其BLOB值,再借助`UTL_FILE` API完成实际的数据写出动作。 #### 注意事项 - **安全性**:由于动态SQL语句的存在,在调用本函数前应仔细验证输入变量的有效性和合法性以防SQL注入攻击。 - **性能优化**:对于特别巨大的BLOB对象可能考虑分块传输而非一次性加载整个内容于内存之中。 - **错误处理**:增加了异常捕获逻辑确保即使发生问题也能正常关闭已打开资源防止泄露。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值