爲了解決大文件上傳 (PHP上傳最大限制2GB)php
同時爲了解決文件上傳是對服務器形成的壓力css
能夠經過分段上傳解決這個問題,這得益於HTML5開發的file APIhtml
前臺代碼:jquery
引用了進度條插件myProgress.jsajax
<link href="__PUBLIC__/admin/css/myProgress.css" rel="stylesheet"> <script src="__PUBLIC__/admin/js/jquery.myProgress.js"></script> <div> <div> <form id="myForm"> <div> //上傳文件時由用戶指定文件名 <label for="FileName">File Name</label> <input type="text" name="title" class="form-control" id="FileName"> </div> <div> <label for="myFile">Chose File</label> <input type="file" id="myFile"> <div class="progress-out" id="progress"> <div class="percent-show"><span>0</span>%</div> <div class="progress-in"></div> </div> </div> </form> <button type="button" class="btn btn-default" id="btn">Submit</button> </div> </div> <script> //初始化上傳 function initUpload() { var chunk = 1000*1024; //每片大小 var input = document.getElementById("myFile"); //input file input.onchange = function (e) { //得到上傳的文件 var file = this.files[0]; //若是大於指定大小 提示錯誤 if(file.size > 1*1024*1024*1024){ $('#help_msg').removeClass('help-block').addClass('error-block'); return ; }else{ $('#help_msg').css('display','none'); } // 開啓進度條 $("#progress").css('display','block'); $("#progress").myProgress({speed: 1000, percent: 0, width: "200px", height: "12px"}); var query = {}; var chunks = []; if (!!file) { var start = 0; //文件分片 for (var i = 0; i < Math.ceil(file.size / chunk); i++) { //最後一段取文件的真實大小 var end = 0; if(i == (Math.ceil(file.size / chunk)-1)){ end = file.size; }else{ end = start + chunk; } chunks[i] = file.slice(start , end); start = end; } // 採用post方法上傳文件 // url query上拼接如下參數,用於記錄上傳偏移 // post body中存放本次要上傳的二進制數據 query = { fileName : file.name, fileSize: file.size, dataSize: chunk, nextOffset: 0 } upload(chunks, query, successPerUpload); } } } // 執行上傳 function upload(chunks, query, cb) { //對象轉字符串 用&鏈接 var queryStr = Object.getOwnPropertyNames(query).map(key => { return key + "=" + query[key]; }).join("&"); var xhr = new XMLHttpRequest(); xhr.open("POST", "/Shop/index.php/Admin/File/upload_file?" + queryStr); xhr.overrideMimeType("application/octet-stream"); //獲取post body中二進制數據 var index = Math.floor(query.nextOffset / query.dataSize); getFileBinary(chunks[index], function (binary) { if (xhr.sendAsBinary) { xhr.sendAsBinary(binary); } else { xhr.send(binary); } }); xhr.onreadystatechange = function (e) { if (xhr.readyState === 4) { if (xhr.status === 200) { var resp = JSON.parse(xhr.responseText); //經過返回數據更新進度條 var precent = Math.ceil((resp.offset / query.fileSize) * 100); $("#progress").myProgress({speed: 1000, percent: precent, width: "200px", height: "12px"}); // 接口返回nextoffset // resp = { // isFinish:false, // offset:100*1024 // } if (typeof cb === "function") { cb.call(this, resp, chunks, query) } } } } } // 每片上傳成功後執行 function successPerUpload(resp, chunks, query) { if (resp.isFinish === true) { //上傳完成給出提示 $('#help_msg').css('display','block').addClass('error-block').html('success!'); } else { //未上傳完畢 query.nextOffset = resp.offset; upload(chunks, query, successPerUpload); } } // 獲取文件二進制數據 function getFileBinary(file, cb) { var reader = new FileReader(); reader.readAsArrayBuffer(file); reader.onload = function (e) { if (typeof cb === "function") { cb.call(this, this.result); } } } //初始化上傳 initUpload(); //ajax模擬提交表單 $(function(){ $('#btn').click(function(){ var fd = new FormData(document.querySelector('#myForm')); var input = document.getElementById("myFile"); //input file var file = input.files[0]; if(!file){ $('#help_msg').css('display','block').addClass('error-block').html('please chose the file !'); return ; } fd.append('FileName',file.name); fd.append('size',file.size); $.ajax({ url : "/Shop/index.php/Admin/File/add", type: "POST", async : true, data: fd, processData: false, // 不處理數據 contentType: false, // 不設置內容類型 success : function(result){ console.log(result); if(result.res == 1){ window.location.href = "http://localhost:8080/Shop/index.php/Admin/File/index"; }else{ $("#help_msg_1").css('display','block').html('upload faild ' + $result.msg); } } }); }) }) </script>
後臺PHP代碼 數據庫
public function add(){ if($_POST){ $Attach = D('Attachment'); $file_path = './Upload/File/'.$_POST['FileName']; //若是是win系統將文件名改爲GBK編碼 if(strtoupper(substr(PHP_OS, 0, 3)) === 'WIN'){ $file_path = iconv('UTF-8', 'GBK', $file_path); } if(file_exists($file_path)){ //得到拓展名 $ext = strtolower(trim(substr(strrchr($_POST['FileName'], '.'), 1))); //生成新的文件名 $url = './Upload/File/'.date("Ymdhms").rand(1000,9999).'.'.$ext; $_POST['url'] = $url ; //將上傳的文件更名,將新的路徑存入數據庫 if(rename($file_path, $url)){ $res = $Attach -> add_file($_POST); if($res['res']){ $log['remark'] = session('userinfo')['name'].'在'.date("Y-m-d H:i:s").'上傳了文件'; D('ActionLog') -> add_log($log); $this -> ajaxReturn(array('res' => 1)); }else{ $this -> ajaxReturn(array('res' => 0 , 'msg' => $res['msg'])); } } }else{ $this -> ajaxReturn(array('res' => 0, 'msg' => '上傳文件不存在')); } }else{ $this -> show(); } } //異步分段上傳文件 public function upload_file(){ $path = './Upload/File/'.$_GET['fileName']; if(strtoupper(substr(PHP_OS, 0, 3)) === 'WIN'){ $path = iconv('UTF-8', 'GBK', $path); } if(!file_exists($path)){ $handle = fopen($path, "a+"); fclose($handle); } file_put_contents($path, file_get_contents('php://input'),FILE_APPEND|LOCK_EX); $offset = filesize($path); if( $offset >= $_GET['fileSize'] ){ $this -> ajaxReturn(array('isFinish' => true)); }else{ $this -> ajaxReturn(array('isFinish' => false , 'offset' => $offset)); } }