PHP使用header方式实现文件下载功能
先给大家介绍下php header() 函数
定义和用法
header() 函数向客户端发送原始的 http 报头。
认识到一点很重要,即必须在任何实际的输出被发送之前调用 header() 函数(在 php 4 以及更高的版本中,您可以使用输出缓存来解决此问题):
<html> <?php // 结果出错 // 在调用 header() 之前已存在输出 header('location: http://www.example.com/'); ?>
语法
header(string,replace,http_response_code)
数 | 描述 |
---|---|
string | 必需。规定要发送的报头字符串。 |
replace |
可选。指示该报头是否替换之前的报头,或添加第二个报头。 默认是 true(替换)。false(允许相同类型的多个报头)。 |
http_response_code | 可选。把 http 响应代码强制为指定的值。(php 4 以及更高版本可用) |
php文件下载可以使用http的请求头加上php的io可以实现,很久之前写过这么一个功能,后来代码没了,今天记录一下
1、先看一下一个正常的http请求
http/1.1 200 ok server: tengine content-type: application/octet-stream content-length: 5050697 connection: keep-alive date: thu, 12 oct 2017 11:24:46 gmt accept-ranges: bytes content-disposition: attachment; filename=down/20170928/zjbb_2.9.5.apk expires: thu, 12 oct 2017 11:25:46 gmt cache-control: max-age=60 via: cache25.l2eu6-1[0,200-0,h], cache16.l2eu6-1[16,0], cache8.cn891[0,200-0,h], cache8.cn891[1,0] age: 1733678 x-cache: hit tcp_mem_hit dirn:6:277104755 mlen:-1 x-swift-savetime: sat, 14 oct 2017 00:50:47 gmt x-swift-cachetime: 93312000 timing-allow-origin: * eagleid: b73d0e1c15095411645886178e
2、一些常见的header功能
header('http/1.1 200 ok'); // ok 正常访问 header('http/1.1 404 not found'); //通知浏览器 页面不存在 header('http/1.1 301 moved permanently'); //设置地址被永久的重定向 301 header('location: http://www.test.con/'); //跳转到一个新的地址 header('refresh: 10; url=http://www.test.con/'); //延迟转向 也就是隔几秒跳转 header('x-powered-by: php/7.0.0'); //修改 x-powered-by信息 header('content-language: en'); //文档语言 header('content-length: 1234'); //设置内容长度 header('last-modified: '.gmdate('d, d m y h:i:s', $time).' gmt'); //告诉浏览器最后一次修改时间 header('http/1.1 304 not modified'); //告诉浏览器文档内容没有发生改变 ###内容类型### header('content-type: text/html; charset=utf-8'); //网页编码 header('content-type: text/plain'); //纯文本格式 header('content-type: image/jpeg'); //jpg、jpeg header('content-type: application/zip'); // zip文件 header('content-type: application/pdf'); // pdf文件 header('content-type: audio/mpeg'); // 音频文件 header('content-type: text/css'); //css文件 header('content-type: text/javascript'); //js文件 header('content-type: application/json'); //json header('content-type: application/pdf'); //pdf header('content-type: text/xml'); //xml header('content-type: application/x-shockw**e-flash'); //flash动画 ###### ###声明一个下载的文件### header('content-type: application/octet-stream'); header('content-disposition: attachment; filename="itblog.zip"'); header('content-transfer-encoding: binary'); readfile('test.zip'); ###### ###对当前文档禁用缓存### header('cache-control: no-cache, no-store, max-age=0, must-revalidate'); header('expires: mon, 26 jul 1997 05:00:00 gmt'); ###### ###显示一个需要验证的登陆对话框### header('http/1.1 401 unauthorized'); header('www-authenticate: basic realm="top secret"'); ###### ###声明一个需要下载的xls文件### header('content-disposition: attachment; filename=abc.xlsx'); header('content-type: application/vnd.openxmlformats-officedocument.spreadsheetml.sheet'); header('content-length: '.filesize('./test.xls')); header('content-transfer-encoding: binary'); header('cache-control: must-revalidate'); header('pragma: public'); 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,如果是大文件,可能会有问题
完整代码
<?php ob_clean(); $action = $_get['action']; $filename = base64_decode($action);//传的参数encode了 $filepath = '/data/www/www.test.com/'.$filename; if(!file_exists($filepath)){ exit; } $fp=fopen($filepath,"r"); $filesize=filesize($filepath); header("content-type:application/octet-stream"); header("accept-ranges:bytes"); header("accept-length:".$filesize); header("content-disposition: attachment; filename=".$filename); $buffer=1024; $buffer_count=0; while(!feof($fp)&&$file_size-$buffer_count>0){ $data=fread($fp,$buffer); $buffer_count+=$buffer; echo $data; } fclose($fp); ?>
ps:下面看一段实例代码php如何通过header文件头实现文件下载
具体代码如下所示:
$file = $_get['file']; if(file_exists($file)){ header("content-type:application/octet-stream"); $filename = basename($file); header("content-disposition:attachment;filename = ".$filename); header("accept-ranges:bytes"); header("accept-length:".filesize($file)); readfile($file); }else{ echo "<script>alert('文件不存在')</script>"; }
总结
以上所述是小编给大家介绍的php使用header方式实现文件下载功能,希望对大家有所帮助
上一篇: 关于cookie获取问题
下一篇: 设置matplotlib画图支持中文显示