gmnon.cn-疯狂蹂躏欧美一区二区精品,欧美精品久久久久a,高清在线视频日韩欧美,日韩免费av一区二区

站長資訊網
最全最豐富的資訊網站

Node.js怎么實現分片上傳?方法介紹

Node怎么實現分片上傳?下面本篇文章給大家介紹一下Node.js實現分片上傳的方法,希望對大家有所幫助!

Node.js怎么實現分片上傳?方法介紹

大文件上傳會消耗大量的時間,而且中途有可能上傳失敗。這時我們需要前端和后端配合來解決這個問題。

解決步驟:

  • 文件分片,減少每次請求消耗的時間,如果某次請求失敗可以單獨上傳,而不是從頭開始

  • 通知服務端合并文件分片

  • 控制并發的請求數量,避免瀏覽器內存溢出

  • 當因為網絡或者其他原因導致某次的請求失敗,我們重新發送請求

文件的分片與合并

在JavaScript中,FIle對象是' Blob '對象的子類,該對象包含一個重要的方法slice,通過該方法我們可以這樣分割二進制文件:

<!DOCTYPE html> <html> <head>     <meta charset="UTF-8">     <meta http-equiv="X-UA-Compatible" content="IE=edge">     <meta name="viewport" content="width=device-width, initial-scale=1.0">     <title>Document</title>     <script src="https://cdn.bootcdn.net/ajax/libs/axios/0.24.0/axios.min.js"></script> </head> <body>     <input type="file" multiple="multiple" id="fileInput" />     <button onclick="SliceUpload()">上傳</button>       <script>         function SliceUpload() {             const file = document.getElementById('fileInput').files[0]             if (!file) return              // 文件分片             let size = 1024 * 50; //50KB 50KB Section size             let fileChunks = [];             let index = 0;        //Section num             for (let cur = 0; cur < file.size; cur += size) {                 fileChunks.push({                     hash: index++,                     chunk: file.slice(cur, cur + size),                 });             }              // 上傳分片             const uploadList = fileChunks.map((item, index) => {                 let formData = new FormData();                 formData.append("filename", file.name);                 formData.append("hash", item.hash);                 formData.append("chunk", item.chunk);                 return axios({                     method: "post",                     url: "/upload",                     data: formData,                 });             });             await Promise.all(uploadList);              // 所有分片上傳完成,通知服務器合并分片             await axios({                 method: "get",                 url: "/merge",                 params: {                     filename: file.name,                 },             });             console.log("Upload to complete");         }     </script> </body> </html>

并發控制

如果文件很大,這樣切分的分片會很多,瀏覽器短時間內就會發起大量的請求,可能會導致內存耗盡,所以要進行并發控制。

這里我們結合Promise.race()方法 控制并發請求的數量,避免瀏覽器內存溢出。

// 加入并發控制 async function SliceUpload() {     const file = document.getElementById('fileInput').files[0]     if (!file) return      // 文件分片     let size = 1024 * 50; //50KB 50KB Section size     let fileChunks = [];     let index = 0;        //Section num     for (let cur = 0; cur < file.size; cur += size) {         fileChunks.push({             hash: index++,             chunk: file.slice(cur, cur + size),         });     }      let pool = []; //Concurrent pool     let max = 3; //Maximum concurrency     for (let i = 0; i < fileChunks.length; i++) {         let item = fileChunks[i];         let formData = new FormData();         formData.append("filename", file.name);         formData.append("hash", item.hash);         formData.append("chunk", item.chunk);          // 上傳分片         let task = axios({             method: "post",             url: "/upload",             data: formData,         });         task.then(() => {         // 從并發池中移除已經完成的請求         let index = pool.findIndex((t) => t === task);             pool.splice(index);         });          // 把請求放入并發池中,如果已經達到最大并發量         pool.push(task);         if (pool.length === max) {             //All requests are requested complete             await Promise.race(pool);         }     }      // 所有分片上傳完成,通知服務器合并分片     await axios({         method: "get",         url: "/merge",         params: {             filename: file.name,         },     });     console.log("Upload to complete"); }

使代碼可復用

function SliceUpload() {     const file = document.getElementById('fileInput').files[0]     if (!file) return      // 文件分片     let size = 1024 * 50; // 分片大小設置     let fileChunks = [];     let index = 0;        // 分片序號     for (let cur = 0; cur < file.size; cur += size) {         fileChunks.push({             hash: index++,             chunk: file.slice(cur, cur + size),         });     }      const uploadFileChunks = async function(list){         if(list.length === 0){             // 所有分片上傳完成,通知如無             await axios({                 method: 'get',                 url: '/merge',                 params: {                     filename: file.name                 }             });             console.log('Upload to complete')             return         }          let pool = []       // 并發池         let max = 3         // 最大并發數         let finish = 0      // 完成數量         let failList = []   // 失敗列表         for(let i=0;i<list.length;i++){             let item = list[i]             let formData = new FormData()             formData.append('filename', file.name)             formData.append('hash', item.hash)             formData.append('chunk', item.chunk)                          let task = axios({                 method: 'post',                 url: '/upload',                 data: formData             })              task.then((data)=>{                 // 從并發池中移除已經完成的請求                 let index = pool.findIndex(t=> t===task)                 pool.splice(index)             }).catch(()=>{                 failList.push(item)             }).finally(()=>{                 finish++                 // 如果有失敗的重新上傳                 if(finish===list.length){                     uploadFileChunks(failList)                 }             })             pool.push(task)             if(pool.length === max){                 await Promise.race(pool)             }         }     }      uploadFileChunks(fileChunks) }

服務端接口實現

const express = require('express') const multiparty = require('multiparty') const fs = require('fs') const path = require('path') const { Buffer } = require('buffer') // file path const STATIC_FILES = path.join(__dirname, './static/files') // Temporary path to upload files const STATIC_TEMPORARY = path.join(__dirname, './static/temporary') const server = express() // Static file hosting server.use(express.static(path.join(__dirname, './dist'))) // Interface for uploading slices server.post('/upload', (req, res) => {     const form = new multiparty.Form();     form.parse(req, function(err, fields, files) {         let filename = fields.filename[0]         let hash = fields.hash[0]         let chunk = files.chunk[0]         let dir = `${STATIC_TEMPORARY}/${filename}`         // console.log(filename, hash, chunk)         try {             if (!fs.existsSync(dir)) fs.mkdirSync(dir)             const buffer = fs.readFileSync(chunk.path)             const ws = fs.createWriteStream(`${dir}/${hash}`)             ws.write(buffer)             ws.close()             res.send(`${filename}-${hash} Section uploaded successfully`)         } catch (error) {             console.error(error)             res.status(500).send(`${filename}-${hash} Section uploading failed`)         }     }) })  //Merged slice interface server.get('/merge', async (req, res) => {     const { filename } = req.query     try {         let len = 0         const bufferList = fs.readdirSync(`${STATIC_TEMPORARY}/${filename}`).map((hash,index) => {             const buffer = fs.readFileSync(`${STATIC_TEMPORARY}/${filename}/${index}`)             len += buffer.length             return buffer         });         //Merge files         const buffer = Buffer.concat(bufferList, len);         const ws = fs.createWriteStream(`${STATIC_FILES}/${filename}`)         ws.write(buffer);         ws.close();         res.send(`Section merge completed`);     } catch (error) {         console.error(error);     } })  server.listen(3000, _ => {     console.log('http://localhost:3000/') })

贊(0)
分享到: 更多 (0)
網站地圖   滬ICP備18035694號-2    滬公網安備31011702889846號
gmnon.cn-疯狂蹂躏欧美一区二区精品,欧美精品久久久久a,高清在线视频日韩欧美,日韩免费av一区二区
大陆极品少妇内射aaaaa| 亚洲男人天堂2021| 国内外成人免费在线视频| 日韩精品在线中文字幕| 色婷婷777777仙踪林| 一级做a免费视频| 一区二区三区日韩视频| 国产高清av片| 99在线精品免费视频| 免费无码不卡视频在线观看| 你真棒插曲来救救我在线观看| 日韩精品在线中文字幕| 免费网站在线观看视频| 日韩精品一区二区三区四| 日本十八禁视频无遮挡| 午夜免费福利小电影| 黄色免费视频大全| 男人的天堂日韩| 97超碰人人爱| 黄色片一级视频| 亚洲高清视频免费| 国产一二三四区在线观看| 人妻互换免费中文字幕| 六月丁香激情网| 午夜激情福利在线| 一区二区久久精品| 蜜臀av性久久久久蜜臀av| 激情综合网婷婷| 日本黄色福利视频| 在线视频一二三区| 欧美 国产 综合| 五月天婷婷亚洲| 男人添女荫道口女人有什么感觉| 国产男女无遮挡| 丝袜人妻一区二区三区| 成人免费无码av| 亚洲成熟丰满熟妇高潮xxxxx| 日韩 欧美 高清| 国内外成人激情免费视频| 成熟了的熟妇毛茸茸| 日韩av.com| 日本www在线播放| 超碰91在线播放| 免费在线观看的av网站| 中文字幕12页| 美女一区二区三区视频| 日韩人妻无码精品久久久不卡| 啊啊啊国产视频| 免费超爽大片黄| 日本三级黄色网址| 欧美成人高潮一二区在线看| 日韩av影视大全| 黑人粗进入欧美aaaaa| 日韩精品视频在线观看视频 | avove在线观看| www.com黄色片| 日日躁夜夜躁aaaabbbb| 成熟丰满熟妇高潮xxxxx视频| 999在线精品视频| 538在线视频观看| 嫩草av久久伊人妇女超级a| 成人在线播放网址| 最新中文字幕久久| 欧洲在线免费视频| 在线观看免费视频高清游戏推荐| 国产极品美女高潮无套久久久| 欧美久久久久久久久久久久久久| 日本高清免费观看| 午夜不卡福利视频| 亚洲图片 自拍偷拍| 日韩av自拍偷拍| 少妇熟女一区二区| 日本一道在线观看| 欧美亚洲黄色片| 麻豆tv在线播放| 久久久999视频| 亚洲人成色77777| 久久精品影视大全| 999这里有精品| 99热都是精品| 成人免费观看cn| 亚洲熟妇av一区二区三区| 人妻无码视频一区二区三区| 欧美 国产 日本| 麻豆三级在线观看| 99精品999| 国产九色porny| 亚洲一区二区三区四区五区xx| 中文字幕第38页| 国产大尺度在线观看| 男人用嘴添女人下身免费视频| 精品一卡二卡三卡| 久热精品在线观看视频| 一本大道东京热无码aⅴ| 久久久久久www| 国产免费又粗又猛又爽| 麻豆传媒网站在线观看| 欧美牲交a欧美牲交aⅴ免费真 | 国产三级三级看三级| 日本高清免费观看| 99爱视频在线| 亚洲欧美日韩一二三区| 成 年 人 黄 色 大 片大 全| 国产成人黄色网址| 久久久天堂国产精品| 日日噜噜噜噜久久久精品毛片| 爽爽爽在线观看| 日本黄网站免费| 法国空姐在线观看免费| aaa毛片在线观看| 免费人成在线观看视频播放| 爆乳熟妇一区二区三区霸乳| 丁香婷婷综合激情| 天天摸天天舔天天操| 116极品美女午夜一级| 久久av秘一区二区三区| 蜜臀av免费观看| 亚洲熟妇av一区二区三区漫画| 国产精品免费看久久久无码| 亚欧美在线观看| 黄色三级视频片| 黄色片一级视频| 欧美 日韩 国产 高清| 大桥未久一区二区| 成人不卡免费视频| 特级丰满少妇一级| 欧美在线观看视频网站| 日韩av三级在线| 国产精品www在线观看| 五月天色婷婷综合| 91香蕉国产线在线观看| 亚洲最大天堂网| 亚洲天堂av一区二区| 国产精品区在线| 日日躁夜夜躁aaaabbbb| 啊啊啊国产视频| 99sesese| 国产xxxxhd| 日本a在线天堂| 日本一级黄视频| 日韩一级性生活片| 国产精品va无码一区二区| 欧美精品一区免费| 亚洲自偷自拍熟女另类| 欧美成人黑人猛交| 别急慢慢来1978如如2| 午夜精品在线免费观看| 污污动漫在线观看| 久久久久久久久久久久久久久国产| 日本人69视频| 特级毛片在线免费观看| 91.com在线| 欧美 日韩 激情| 久久精品影视大全| 中文字幕在线视频一区二区| 日本中文字幕在线不卡| 久久国产午夜精品理论片最新版本| 国产美女无遮挡网站| 色91精品久久久久久久久| 天天干天天色天天爽| 免费看又黄又无码的网站| www.99av.com| 国产高清不卡无码视频| 成人观看免费完整观看| 国产九九热视频| 99在线观看视频免费| 国产aaaaa毛片| 国产一级片91| youjizzxxxx18| 男人c女人视频| 超碰在线公开97| 免费视频爱爱太爽了| 波多野结衣xxxx| 日韩小视频在线播放| 999热精品视频| 免费激情视频在线观看| 男同互操gay射视频在线看| 久久久久久久久久久99| 一级黄色片在线免费观看| 国产乱子伦农村叉叉叉| 五月天色婷婷综合| 日日摸天天爽天天爽视频| 黄色高清视频网站| 在线观看免费污视频| 国产精品无码av在线播放| 国产人妻互换一区二区| 激情黄色小视频| 黄色国产精品视频| 国产欧美日韩小视频| 99久久久无码国产精品性色戒| 成人免费毛片播放| 免费毛片小视频| 免费毛片网站在线观看| 喜爱夜蒲2在线| 国产精品无码乱伦| √天堂资源在线| 三级av免费看| 日韩av影视大全| 免费看av软件| av动漫在线免费观看|