溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶(hù)服務(wù)條款》

利用HTML5分片上傳超大文件

發(fā)布時(shí)間:2020-03-03 13:26:16 來(lái)源:網(wǎng)絡(luò) 閱讀:21044 作者:BoyTNT 欄目:移動(dòng)開(kāi)發(fā)

在網(wǎng)頁(yè)中直接上傳大文件一直是個(gè)比較頭疼的問(wèn)題,主要面臨的問(wèn)題一般包括兩類(lèi):一是上傳時(shí)間長(zhǎng)中途一旦出錯(cuò)會(huì)導(dǎo)致前功盡棄;二是服務(wù)端配置復(fù)雜,要考慮接收超大表單和超時(shí)問(wèn)題,如果是托管主機(jī)沒(méi)準(zhǔn)還改不了配置,默認(rèn)只能接收小于4MB的附件。


比較理想的方案是能夠把大文件分片,一片一片的傳到服務(wù)端,再由服務(wù)端合并。這么做的好處在于一旦上傳失敗只是損失一個(gè)分片而已,不用整個(gè)文件重傳,而且每個(gè)分片的大小可以控制在4MB以?xún)?nèi),服務(wù)端不用做任何設(shè)置就可適應(yīng)。


常用的解決方案是RIA,以flex為例,通常是利用FileReference.load方法加載文件得到ByteArray,然后分片構(gòu)造表單(flash的高版本不允許直接訪(fǎng)問(wèn)文件)。不過(guò)這個(gè)load方法只能加載較小的文件,大約不超過(guò)300MB,因此適用性不是很強(qiáng)。


好在現(xiàn)在有了HTML5,我們可以直接構(gòu)造分片了,這是一個(gè)非常喜人的進(jìn)步,只可惜目前適用面不廣(IE啊IE,真是恨你恨得牙癢癢)。


言歸正傳,來(lái)看一個(gè)DEMO吧,基于A(yíng)SP.Net MVC3,只是示例,很多問(wèn)題做了簡(jiǎn)化處理。


主要是客戶(hù)端,新特性都體現(xiàn)在這里:

<%@ Page Language="C#" Inherits="System.Web.Mvc.ViewPage<dynamic>" %>
<!DOCTYPE html>
<html lang="zh-CN">
<head>
    <meta charset="utf-8">
    <title>HTML5大文件分片上傳示例</title>
    <script src="../Scripts/jquery-1.11.1.min.js"></script>
    <script>
    var page = {
        init: function(){
            $("#upload").click($.proxy(this.upload, this));
        },
        
        upload: function(){
            var file = $("#file")[0].files[0],  //文件對(duì)象
                name = file.name,        //文件名
                size = file.size,        //總大小
                succeed = 0;
                 
            var shardSize = 2 * 1024 * 1024,     //以2MB為一個(gè)分片
                shardCount = Math.ceil(size / shardSize);   //總片數(shù)
                 
            for(var i = 0;i < shardCount;++i){
                //計(jì)算每一片的起始與結(jié)束位置
                var start = i * shardSize,
                    end = Math.min(size, start + shardSize);

                //構(gòu)造一個(gè)表單,F(xiàn)ormData是HTML5新增的
                var form = new FormData();
                form.append("data", file.slice(start,end));  //slice方法用于切出文件的一部分
                form.append("name", name);
                form.append("total", shardCount);   //總片數(shù)
                form.append("index", i + 1);        //當(dāng)前是第幾片
                
                //Ajax提交
                $.ajax({
                    url: "../File/Upload",
                    type: "POST",
                    data: form,
                    async: true,         //異步
                    processData: false,  //很重要,告訴jquery不要對(duì)form進(jìn)行處理
                    contentType: false,  //很重要,指定為false才能形成正確的Content-Type
                    success: function(){
                        ++succeed;
                        $("#output").text(succeed + " / " + shardCount);
                    }
                });
            }
        }
    };
    $(function(){
        page.init();
    });
    </script>
</head>
<body>
    <input type="file" id="file" />
    <button id="upload">上傳</button>
    <span id="output" >等待</span>
</body>
</html>


這里的slice方法和FormData都是html5之前不存在的。通過(guò)這樣的方法,我們的表單構(gòu)造出來(lái)是這樣的,抓包看看:

利用HTML5分片上傳超大文件

可以看到構(gòu)造出來(lái)的Content-Type是multipart/form-data,也就是符合RFC標(biāo)準(zhǔn)的那個(gè)最傳統(tǒng)的文件上傳表單。另外我們同時(shí)傳輸?shù)膎ame、total等屬性也都在表單里。


然后是服務(wù)端,沒(méi)什么新鮮的,完全是在接收一個(gè)普通的文件:

[HttpPost]
public ActionResult Upload()
{
    //從Request中取參數(shù),注意上傳的文件在Requst.Files中
    string name = Request["name"];
    int total = Convert.ToInt32(Request["total"]);
    int index = Convert.ToInt32(Request["index"]);
    var data = Request.Files["data"];
    
    //保存一個(gè)分片到磁盤(pán)上
    string dir = Server.MapPath("~/Upload");
    string file = Path.Combine(dir, name + "_" + index);
    data.SaveAs(file);
    
    //如果已經(jīng)是最后一個(gè)分片,組合
    //當(dāng)然你也可以用其它方法比如接收每個(gè)分片時(shí)直接寫(xiě)到最終文件的相應(yīng)位置上,但要控制好并發(fā)防止文件鎖沖突
    if(index == total)
    {
        file = Path.Combine(dir, name);
        var fs = new FileStream(file, FileMode.Create);
        for(int i = 1;i <= total;++i)
        {
            string part = Path.Combine(dir, name + "_" + i);
            var bytes = System.IO.File.ReadAllBytes(part);
            fs.Write(bytes, 0, bytes.Length);
            bytes = null;
            System.IO.File.Delete(part);
        }
        fs.Close();
    }
    
    //返回是否成功,此處做了簡(jiǎn)化處理
    return Json(new { Error = 0 });
}


上面的DEMO很多問(wèn)題是簡(jiǎn)化處理的,比如沒(méi)做什么異常處理,客戶(hù)端也沒(méi)有判斷服務(wù)端是否出錯(cuò)重試一類(lèi)的,各位可以自己完善。


在上面的基礎(chǔ)上,我們可以做很多功能上的擴(kuò)展,比如我們可以控制所有分片是順序上傳還是并發(fā)上傳,以適用不同應(yīng)用。再比如我們可以在整體文件上傳前以及分片上傳前都先計(jì)算一下相應(yīng)的HASH,發(fā)個(gè)請(qǐng)求詢(xún)問(wèn)服務(wù)器文件是否已存在,如果存在就不要重復(fù)上傳了,這樣就實(shí)現(xiàn)了“極速上傳”以及“斷點(diǎn)續(xù)傳”。



向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀(guān)點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI