这是一个将文件发送到客户端的函数 - 它可能看起来比必要时更复杂,但与更简单的文件发送函数相比,它具有一些优点
- 可以处理大文件,并且每次传输只使用 8KB 的缓冲区。
- 如果客户端断开连接,则停止传输(与许多继续读取和缓冲整个文件的脚本不同,这会浪费宝贵的资源),但不会停止脚本
- 如果传输完成,则返回 TRUE,如果客户端在完成下载之前断开连接,则返回 FALSE - 您通常需要此功能,以便您可以正确记录下载。
- 发送多个标头,包括确保在任何浏览器/代理上最多缓存 2 小时的标头,以及大多数人似乎忘记的“Content-Length”。
(在 PHP4.3.x 下的 Linux(Apache)和 Windows(IIS5/6)上测试)
请注意,将从中提取受保护文件的文件夹在此函数中设置为一个常量 (/protected) ... 现在是该函数
<?php
function send_file($name) {
ob_end_clean();
$path = "protected/".$name;
if (!is_file($path) or connection_status()!=0) return(FALSE);
header("Cache-Control: no-store, no-cache, must-revalidate");
header("Cache-Control: post-check=0, pre-check=0", false);
header("Pragma: no-cache");
header("Expires: ".gmdate("D, d M Y H:i:s", mktime(date("H")+2, date("i"), date("s"), date("m"), date("d"), date("Y")))." GMT");
header("Last-Modified: ".gmdate("D, d M Y H:i:s")." GMT");
header("Content-Type: application/octet-stream");
header("Content-Length: ".(string)(filesize($path)));
header("Content-Disposition: inline; filename=$name");
header("Content-Transfer-Encoding: binary\n");
if ($file = fopen($path, 'rb')) {
while(!feof($file) and (connection_status()==0)) {
print(fread($file, 1024*8));
flush();
}
fclose($file);
}
return((connection_status()==0) and !connection_aborted());
}
?>
下面是一个使用该函数的例子
<?php
if (!send_file("platinumdemo.zip")) {
die ("文件传输失败");
} else {
}
?>
此致,
Rasmus Schultz