PHP使用header方式实现文件下载功能

这篇文章主要介绍了PHP使用header方式实现文件下载功能,非常不错,具有参考借鉴价值,需要的朋友可以参考下。

先给大家介绍下PHP header() 函数

定义和用法

header() 函数向客户端发送原始的 HTTP 报头。

认识到一点很重要,即必须在任何实际的输出被发送之前调用 header() 函数(在 PHP 4 以及更高的版本中,您可以使用输出缓存来解决此问题):

  1. <html>
  2. <?php
  3. // 结果出错
  4. // 在调用 header() 之前已存在输出
  5. header('Location: http://www.example.com/');
  6. ?>

语法

header(string,replace,http_response_code)

数 描述

string 必需。规定要发送的报头字符串。

replace

可选。指示该报头是否替换之前的报头,或添加第二个报头。

默认是 true(替换)。false(允许相同类型的多个报头)。

http_response_code 可选。把 HTTP 响应代码强制为指定的值。(PHP 4 以及更高版本可用)

php文件下载可以使用http的请求头加上php的IO可以实现,很久之前写过这么一个功能,后来代码没了,今天记录一下

1、先看一下一个正常的http请求

  1. HTTP/1.1 200 OK
  2. Server: Tengine
  3. Content-Type: application/octet-stream
  4. Content-Length: 5050697
  5. Connection: keep-alive
  6. Date: Thu, 12 Oct 2017 11:24:46 GMT
  7. Accept-Ranges: bytes
  8. Content-Disposition: attachment; filename=down/20170928/zjbb_2.9.5.apk
  9. Expires: Thu, 12 Oct 2017 11:25:46 GMT
  10. Cache-Control: max-age=60
  11. Via: cache25.l2eu6-1[0,200-0,H], cache16.l2eu6-1[16,0], cache8.cn891[0,200-0,H], cache8.cn891[1,0]
  12. Age: 1733678
  13. X-Cache: HIT TCP_MEM_HIT dirn:6:277104755 mlen:-1
  14. X-Swift-SaveTime: Sat, 14 Oct 2017 00:50:47 GMT
  15. X-Swift-CacheTime: 93312000
  16. Timing-Allow-Origin: *
  17. EagleId: b73d0e1c15095411645886178e

2、一些常见的header功能

  1. header('HTTP/1.1 200 OK'); // ok 正常访问
  2. header('HTTP/1.1 404 Not Found'); //通知浏览器 页面不存在
  3. header('HTTP/1.1 301 Moved Permanently'); //设置地址被永久的重定向 301
  4. header('Location: http://www.test.con/'); //跳转到一个新的地址
  5. header('Refresh: 10; url=http://www.test.con/'); //延迟转向 也就是隔几秒跳转
  6. header('X-Powered-By: PHP/7.0.0'); //修改 X-Powered-By信息
  7. header('Content-language: en'); //文档语言
  8. header('Content-Length: 1234'); //设置内容长度
  9. header('Last-Modified: '.gmdate('D, d M Y H:i:s', $time).' GMT'); //告诉浏览器最后一次修改时间
  10. header('HTTP/1.1 304 Not Modified'); //告诉浏览器文档内容没有发生改变
  11. ###内容类型###
  12. header('Content-Type: text/html; charset=utf-8'); //网页编码
  13. header('Content-Type: text/plain'); //纯文本格式
  14. header('Content-Type: image/jpeg'); //JPG、JPEG
  15. header('Content-Type: application/zip'); // ZIP文件
  16. header('Content-Type: application/pdf'); // PDF文件
  17. header('Content-Type: audio/mpeg'); // 音频文件
  18. header('Content-type: text/css'); //css文件
  19. header('Content-type: text/javascript'); //js文件
  20. header('Content-type: application/json'); //json
  21. header('Content-type: application/pdf'); //pdf
  22. header('Content-type: text/xml'); //xml
  23. header('Content-Type: application/x-shockw**e-flash'); //Flash动画
  24. ######
  25. ###声明一个下载的文件###
  26. header('Content-Type: application/octet-stream');
  27. header('Content-Disposition: attachment; filename="ITblog.zip"');
  28. header('Content-Transfer-Encoding: binary');
  29. readfile('test.zip');
  30. ######
  31. ###对当前文档禁用缓存###
  32. header('Cache-Control: no-cache, no-store, max-age=0, must-revalidate');
  33. header('Expires: Mon, 26 Jul 1997 05:00:00 GMT');
  34. ######
  35. ###显示一个需要验证的登陆对话框###
  36. header('HTTP/1.1 401 Unauthorized');
  37. header('WWW-Authenticate: Basic realm="Top Secret"');
  38. ######
  39. ###声明一个需要下载的xls文件###
  40. header('Content-Disposition: attachment; filename=abc.xlsx');
  41. header('Content-Type: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet');
  42. header('Content-Length: '.filesize('./test.xls'));
  43. header('Content-Transfer-Encoding: binary');
  44. header('Cache-Control: must-revalidate');
  45. header('Pragma: public');
  46. readfile('./test.xls');

3、看下下载所要用的的请求头

header("Content-type:application/octet-stream");

header("Accept-Ranges:bytes");

header("Accept-Length:".$file_Size);

header("Content-Disposition: attachment; filename=".$filename);

content-type:文件类型

Accept-Ranges:表示接收数据的类型或者范围,图片属于二进制的东西所以需要使用字节的方式传输

Accept-Length:表示接收的文件大小,php文件下载需要告诉浏览器下载的文件有多大

Content-Disposition:附件只需要把文件名给过去就可以,这个名称就是下载时显示的文件名称

4、php的文件操作出现的比较早,文件名是中文的时候需要注意转码

$filename=iconv("UTF-8","GB2312",$filename);

5、php的文件下载机制是首先nginx把文件信息读入服务器内存,然后使用请求头把文件二进制信息通过浏览器传给客户端

feof用来判断文件是否已经读到了末尾,fread用来把文件读入缓冲区,缓冲区的大小是1024,一边读取一边把数据输出到浏览器。为了下载的安全性每次读数据都进行字节的计数。文件读取完毕后关闭输入流

注意:

a、如果运行的过程中出现问题,可以清空(擦掉)输出缓冲区,使用下面的代码即可

ob_clean();

b、很多人喜欢用readfile,如果是大文件,可能会有问题

完整代码

  1. <?php
  2. ob_clean();
  3. $action = $_GET['action'];
  4. $filename = base64_decode($action);//传的参数encode了
  5. $filepath = '/data/www/www.test.com/'.$filename;
  6. if(!file_exists($filepath)){
  7. exit;
  8. }
  9. $fp=fopen($filepath,"r");
  10. $filesize=filesize($filepath);
  11. header("Content-type:application/octet-stream");
  12. header("Accept-Ranges:bytes");
  13. header("Accept-Length:".$filesize);
  14. header("Content-Disposition: attachment; filename=".$filename);
  15. $buffer=1024;
  16. $buffer_count=0;
  17. while(!feof($fp)&&$file_Size-$buffer_count>0){
  18. $data=fread($fp,$buffer);
  19. $buffer_count+=$buffer;
  20. echo $data;
  21. }
  22. fclose($fp);
  23. ?>

PS:下面看一段实例代码php如何通过header文件头实现文件下载

具体代码如下所示:

  1. $file = $_GET['file'];
  2. if(file_exists($file)){
  3. header("Content-type:application/octet-stream");
  4. $filename = basename($file);
  5. header("Content-Disposition:attachment;filename = ".$filename);
  6. header("Accept-ranges:bytes");
  7. header("Accept-length:".filesize($file));
  8. readfile($file);
  9. }else{
  10. echo "<script>alert('文件不存在')</script>";
  11. }