JS 異步分段上傳文件

爲了解決大文件上傳 (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));
        }
    }
相關文章
相關標籤/搜索