PostgreSQL中的bytea字段类型可以以二进制的形式存储数据,这样做的好处就是可以将原本存储在网站目录下的文件存储到数据库中,坏处就是如果文件过多、过大的话,就会导致数据库的数据量大大增加,备份和恢复的时候就会浪费大量的时间,而且数据也有可能会出错。个人觉得,在文件量小的情况下,使用这种存储方式还是很方便的。
言归正传,下面介绍一下使用bytea字段存读取文件的具体实现方法。首先是文件存储于存储于bytea字段的方法,主要用到的就是PHP中的pg_escape_bytea方法,代码如下:
<?php $dbconn = pg_connect("host='localhost' dbname='dbname' user='user' password='password' port='port'") OR DIE('Could not connect:' . pg_last_error()); $fileUrl = iconv("utf-8", "gbk", $fileUrl);//$fileName为文件路径地址,汉字可能会乱码,此处处理一下 $fileContents = file_get_contents($fileUrl); $escapeBytea = pg_escape_bytea($fileContents);//转义bytea数据类型的二进制字符串 $insertStr = "INSERT INTO tableName(id, contents) VALUES(DEFAULT, '{$escapeBytea}')"; pg_query($dbconn, $insertStr); ?>
从bytea字段中还原文件用到的是PHP中的pg_unescape_bytea方法,实现代码如下:
<?php $dbconn = pg_connect("host='localhost' dbname='dbname' user='user' password='password' port='port'") OR DIE('Could not connect: ' . pg_last_error()); $selectStr = "SELECT contents FROM tableName WHERE id = " . $fileId;//$fileId为文件id $query = pg_query($dbconn, $selectStr); while($row = pg_fetch_array($query, null, PGSQL_ASSOC)){ $escapeBytea = $row['contents']; $fileContents = pg_unescape_bytea($escapeBytea); //获得二进制数据 file_put_contents($fileName, $fileContents); //$fileName为带有后缀名的文件名,如hello.pdf } ?>
导出文件后,如果二进制数据转码错误就会出现文件打不开的现象,比如错误的PDF文件打开时弹出错误如下图:
这种错误在数据库迁移时特别容易出现(本人是从PostgreSQL 8.4迁移到9.1),解决的方法是修改PostgreSQL的配置文件
postgresql.conf,将bytea_output的输出类型设置为转义类型(escape)输出,即bytea_output = 'escape'(如果前面有#,删除开启配置),然后reload一下PostgreSQL的配置使修改生效,这样二进制数据就可以正常解码并输出到文件。