利用HTML5分片上傳超大文件-創(chuàng)新互聯(lián)

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

成都創(chuàng)新互聯(lián)成立以來不斷整合自身及行業(yè)資源、不斷突破觀念以使企業(yè)策略得到完善和成熟,建立了一套“以技術(shù)為基點,以客戶需求中心、市場為導(dǎo)向”的快速反應(yīng)體系。對公司的主營項目,如中高端企業(yè)網(wǎng)站企劃 / 設(shè)計、行業(yè) / 企業(yè)門戶設(shè)計推廣、行業(yè)門戶平臺運營、成都App定制開發(fā)、移動網(wǎng)站建設(shè)、微信網(wǎng)站制作、軟件開發(fā)、綿陽電信機(jī)房機(jī)柜租用等實行標(biāo)準(zhǔn)化操作,讓客戶可以直觀的預(yù)知到從成都創(chuàng)新互聯(lián)可以獲得的服務(wù)效果。

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

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

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

言歸正傳,來看一個DEMO吧,基于ASP.Net MVC3,只是示例,很多問題做了簡化處理。

主要是客戶端,新特性都體現(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],  //文件對象                 name = file.name,        //文件名                 size = file.size,        //總大小                 succeed = 0;                               var shardSize = 2 * 1024 * 1024,     //以2MB為一個分片                 shardCount = Math.ceil(size / shardSize);   //總片數(shù)                               for(var i = 0;i < shardCount;++i){                 //計算每一片的起始與結(jié)束位置                 var start = i * shardSize,                     end = Math.min(size, start + shardSize);                 //構(gòu)造一個表單,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不要對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之前不存在的。通過這樣的方法,我們的表單構(gòu)造出來是這樣的,抓包看看:

利用HTML5分片上傳超大文件

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

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

[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"];          //保存一個分片到磁盤上     string dir = Server.MapPath("~/Upload");     string file = Path.Combine(dir, name + "_" + index);     data.SaveAs(file);          //如果已經(jīng)是最后一個分片,組合     //當(dāng)然你也可以用其它方法比如接收每個分片時直接寫到最終文件的相應(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();     }          //返回是否成功,此處做了簡化處理     return Json(new { Error = 0 }); }

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

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

另外有需要云服務(wù)器可以了解下創(chuàng)新互聯(lián)cdcxhl.cn,海內(nèi)外云服務(wù)器15元起步,三天無理由+7*72小時售后在線,公司持有idc許可證,提供“云服務(wù)器、裸金屬服務(wù)器、高防服務(wù)器、香港服務(wù)器、美國服務(wù)器、虛擬主機(jī)、免備案服務(wù)器”等云主機(jī)租用服務(wù)以及企業(yè)上云的綜合解決方案,具有“安全穩(wěn)定、簡單易用、服務(wù)可用性高、性價比高”等特點與優(yōu)勢,專為企業(yè)上云打造定制,能夠滿足用戶豐富、多元化的應(yīng)用場景需求。

當(dāng)前文章:利用HTML5分片上傳超大文件-創(chuàng)新互聯(lián)
轉(zhuǎn)載源于:http://bm7419.com/article34/ggsse.html

成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供做網(wǎng)站、企業(yè)網(wǎng)站制作網(wǎng)站建設(shè)、響應(yīng)式網(wǎng)站網(wǎng)頁設(shè)計公司、標(biāo)簽優(yōu)化

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

網(wǎng)站托管運營