php实现断点续传大文件示例代码
吾爱主题
阅读:116
2021-10-18 14:48:00
评论:0
一、断点续传原理
所谓断点续传,也就是要从文件已经下载的地方开始继续下载。在以前版本的 HTTP 协议是不支持断点的,HTTP/1.1 开始就支持了。一般断点下载时才用到 Range 和 Content-Range 实体头。
不使用断点续传
?1 2 3 4 5 6 7 | get /down.zip http/1.1 accept: image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, application/vnd.ms- excel, application/msword, application/vnd.ms-powerpoint, */* accept-language: zh-cn accept-encoding: gzip, deflate user-agent: mozilla/4.0 (compatible; msie 5.01; windows nt 5.0) connection: keep-alive |
服务器收到请求后,按要求寻找请求的文件,提取文件的信息,然后返回给浏览器,返回信息如下:
?1 2 3 4 5 6 7 8 | HTTP/1.1 200 Ok content-length=106786028 accept-ranges=bytes date =mon, 30 apr 2001 12:56:11 gmt etag=w/ "02ca57e173c11:95b" content-type=application/octet-stream server=microsoft-iis/5.0 last-modified=mon, 30 apr 2001 12:56:11 gmt |
使用断点续传
?1 2 3 4 | GET /down.zip HTTP/1.0 User-Agent: NetFox RANGE: bytes=2000070- Accept: text/html, image/gif, image/jpeg, *; q=.2, */*; q=.2 |
多了这么一行Range: bytes=2000070-
这一行的意思就是告诉服务器down.zip这个文件从2000070字节开始传,前面的字节不用传了。
Range的完整格式是:
1 2 3 | Range: bytes=startOffset-targetOffset/sum [表示从startOffset读取,一直读取到targetOffset位置,读取总数为sum直接] Range: bytes=startOffset-targetOffset [字节总数也可以去掉] |
服务器收到这个请求以后,返回的信息如下:
?1 2 3 4 5 6 7 8 | HTTP/1.1 206 Partial Content content-length=106786028 content-range=bytes 2000070-106786027/106786028 date =mon, 30 apr 2001 12:55:20 gmt etag=w/ "02ca57e173c11:95b" content-type=application/octet-stream server=microsoft-iis/5.0 last-modified=mon, 30 apr 2001 12:55:20 gmt |
和前面服务器返回的信息比较一下,就会发现增加了一行:
?1 | Content-Range=bytes 2000070-106786027/106786028 |
返回的代码也改为206了,而不再是200了。
?1 | HTTP/1.1 206 Partial Content |
知道了以上原理,就可以进行断点续传的编程了。
二、PHP实现
?1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 | /** php下载类,支持断点续传 * download: 下载文件 * setSpeed: 设置下载速度 * getRange: 获取header中Range */ class FileDownload{ /** 下载 * @param String $file 要下载的文件路径 * @param String $name 文件名称,为空则与下载的文件名称一样 * @param boolean $reload 是否开启断点续传 */ public function download( $file , $name = '' , $reload =false){ $fp = @ fopen ( $file , 'rb' ); if ( $fp ){ if ( $name == '' ){ $name = basename ( $file ); } $header_array = get_headers( $file , true); //var_dump($header_array);die; // 下载本地文件,获取文件大小 if (! $header_array ) { $file_size = filesize ( $file ); } else { $file_size = $header_array [ 'Content-Length' ]; } $ranges = $this ->getRange( $file_size ); $ua = $_SERVER [ "HTTP_USER_AGENT" ]; //判断是什么类型浏览器 header( 'cache-control:public' ); header( 'content-type:application/octet-stream' ); $encoded_filename = urlencode( $name ); $encoded_filename = str_replace ( "+" , "%20" , $encoded_filename ); //解决下载文件名乱码 if (preg_match( "/MSIE/" , $ua ) || preg_match( "/Trident/" , $ua ) ){ header( 'Content-Disposition: attachment; filename="' . $encoded_filename . '"' ); } else if (preg_match( "/Firefox/" , $ua )) { header( 'Content-Disposition: attachment; filename*="utf8\'\'' . $name . '"' ); } else if (preg_match( "/Chrome/" , $ua )) { header( 'Content-Disposition: attachment; filename="' . $encoded_filename . '"' ); } else { header( 'Content-Disposition: attachment; filename="' . $name . '"' ); } //header('Content-Disposition: attachment; filename="' . $name . '"'); if ( $reload && $ranges !=null){ // 使用续传 header( 'HTTP/1.1 206 Partial Content' ); header( 'Accept-Ranges:bytes' ); // 剩余长度 header(sprintf( 'content-length:%u' , $ranges [ 'end' ]- $ranges [ 'start' ])); // range信息 header(sprintf( 'content-range:bytes %s-%s/%s' , $ranges [ 'start' ], $ranges [ 'end' ], $file_size )); //file_put_contents('test.log',sprintf('content-length:%u',$ranges['end']-$ranges['start']),FILE_APPEND); // fp指针跳到断点位置 fseek ( $fp , sprintf( '%u' , $ranges [ 'start' ])); } else { file_put_contents ( 'test.log' , '2222' ,FILE_APPEND); header( 'HTTP/1.1 200 OK' ); header( 'content-length:' . $file_size ); } while (! feof ( $fp )){ //echo fread($fp, round($this->_speed*1024,0)); //echo fread($fp, $file_size); echo fread ( $fp , 4096); ob_flush(); } ( $fp !=null) && fclose( $fp ); } else { return '' ; } } /** 设置下载速度 * @param int $speed */ public function setSpeed( $speed ){ if ( is_numeric ( $speed ) && $speed >16 && $speed <4096){ $this ->_speed = $speed ; } } /** 获取header range信息 * @param int $file_size 文件大小 * @return Array */ private function getRange( $file_size ){ //file_put_contents('range.log', json_encode($_SERVER), FILE_APPEND); if (isset( $_SERVER [ 'HTTP_RANGE' ]) && ! empty ( $_SERVER [ 'HTTP_RANGE' ])){ $range = $_SERVER [ 'HTTP_RANGE' ]; $range = preg_replace( '/[\s|,].*/' , '' , $range ); $range = explode ( '-' , substr ( $range , 6)); if ( count ( $range )<2){ $range [1] = $file_size ; } $range = array_combine ( array ( 'start' , 'end' ), $range ); if ( empty ( $range [ 'start' ])){ $range [ 'start' ] = 0; } if ( empty ( $range [ 'end' ])){ $range [ 'end' ] = $file_size ; } return $range ; } return null; } } $obj = new FileDownload(); $obj ->download( 'http://down.golaravel.com/laravel/laravel-master.zip' , '' , true); |
以上就是php实现断点续传大文件示例代码的详细内容,更多关于php 断点续传大文件的资料请关注服务器之家其它相关文章!
原文链接:https://www.cnblogs.com/qianhaihai/p/13154321.html
声明
1.本站遵循行业规范,任何转载的稿件都会明确标注作者和来源;2.本站的原创文章,请转载时务必注明文章作者和来源,不尊重原创的行为我们将追究责任;3.作者投稿可能会经我们编辑修改或补充。