首页 > 其他分享 >大文件分片上传与webWorker多线程执行

大文件分片上传与webWorker多线程执行

时间:2024-12-01 13:30:16浏览次数:7  
标签:function const js 15 webWorker file 分片 new 多线程

前言:前端技术不可避免都要用到文件上传,其中大文件上传是一大难点,需要考虑多种场景;当用户上传大文件的时候,我们需要保证页面的流畅,还要监听上传进度,还有用户如果取消上传后,再度上传相同文件,是否需要从头上传,今天分享一下如何通过js实现大文件分片上传功能,以及webWorker多线程执行分片。

1. html 结构,index.html

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
    <style>
      .ball {
        width: 50px;
        height: 50px;
        border-radius: 50%;
        background: #f40;
        position: fixed;
        bottom: 10px;
        left: 10px;
        animation: move 5s linear infinite alternate;
      }
      @keyframes move {
        to {
          left: calc(100vw - 60px);
        }
      }
    </style>
  </head>
  <body>
    <input type="file" />
    <div class="ball"></div>
    <script src="./main.js" type="module"></script>
  </body>
</html>

代码中除了基础上传功能,还增加了一段动画,用来模拟浏览器渲染时执行js耗时操作是否会造成页面卡顿,事实证明,单线程在计算分片时页面会有明显卡顿,因此采用webWorker多线程。

2. main.js

监听 input 按钮,获取到文件,然后传给 cutFile 函数进行分片。返回分片结果。

import { cutFile } from './cutFile.js';
const inpFile = document.querySelector('input[type="file"]');
inpFile.onchange = async (e) => {
  const file = e.target.files[0];
  const chunks = await cutFile(file);
  console.timeEnd('cutFile');
  console.log(chunks);
};

 3. cutFile.js

切片函数编写,定义 CHUNK_SIZE 每片大小,开启多线程,拿到你电脑内核数。定义 cutFile 方法,传入 file 开始根据内核开始切片。

const CHUNK_SIZE = 1024 * 1024 * 5; // 5MB
const THREAD_COUNT = navigator.hardwareConcurrency || 4;

export function cutFile(file) {
  return new Promise((resolve) => {
    const result = [];
    const chunkCount = Math.ceil(file.size / CHUNK_SIZE);
    const workerChunkCount = Math.ceil(chunkCount / THREAD_COUNT);
    let finishCount = 0;
    for (let i = 0; i < THREAD_COUNT; i++) {
      // 创建一个新的 Worker 线程
      const worker = new Worker('./worker.js', {
        type: 'module',
      });
      // 计算每个线程的开始索引和结束索引
      const startIndex = i * workerChunkCount;
      let endIndex = startIndex + workerChunkCount;
      if (endIndex > chunkCount) {
        endIndex = chunkCount;
      }
      worker.postMessage({
        file,
        CHUNK_SIZE,
        startIndex,
        endIndex,
      });
      worker.onmessage = (e) => {
        // e.data ??
        for (let i = startIndex; i < endIndex; i++) {
          result[i] = e.data[i - startIndex];
        }
        worker.terminate();
        finishCount++;
        if (finishCount === THREAD_COUNT) {
          resolve(result);
        }
      };
    }
  });
}

4. worker.js

定义线程切片方法

import { createChunk } from './createChunk.js';
onmessage = async (e) => {
  const proms = [];
  const { file, CHUNK_SIZE, startIndex, endIndex } = e.data;
  for (let i = startIndex; i < endIndex; i++) {
    proms.push(createChunk(file, i, CHUNK_SIZE));
  }
  const chunks = await Promise.all(proms);
  postMessage(chunks);
};

5. createChunk.js

给每个分片通过 md5 编码设置一个 hash,判断是否已经上传。

import SparkMD5 from './sparkmd5.js';
export function createChunk(file, index, chunkSize) {
  return new Promise((resolve) => {
    const start = index * chunkSize;
    const end = start + chunkSize;
    const spark = new SparkMD5.ArrayBuffer();
    const fileReader = new FileReader();
    fileReader.onload = (e) => {
      spark.append(e.target.result);
      resolve({
        start,
        end,
        index,
        hash: spark.end(),
      });
    };
    fileReader.readAsArrayBuffer(file.slice(start, end));
  });
}

6. sparkmd5.js

用于生成MD5编码的js文件

function factory(undefined){'use strict';var add32=function(a,b){return(a+b)&4294967295},hex_chr=['0','1','2','3','4','5','6','7','8','9','a','b','c','d','e','f',];function cmn(q,a,b,x,s,t){a=add32(add32(a,q),add32(x,t));return add32((a<<s)|(a>>>(32-s)),b)}function md5cycle(x,k){var a=x[0],b=x[1],c=x[2],d=x[3];a+=(((b&c)|(~b&d))+k[0]-680876936)|0;a=(((a<<7)|(a>>>25))+b)|0;d+=(((a&b)|(~a&c))+k[1]-389564586)|0;d=(((d<<12)|(d>>>20))+a)|0;c+=(((d&a)|(~d&b))+k[2]+606105819)|0;c=(((c<<17)|(c>>>15))+d)|0;b+=(((c&d)|(~c&a))+k[3]-1044525330)|0;b=(((b<<22)|(b>>>10))+c)|0;a+=(((b&c)|(~b&d))+k[4]-176418897)|0;a=(((a<<7)|(a>>>25))+b)|0;d+=(((a&b)|(~a&c))+k[5]+1200080426)|0;d=(((d<<12)|(d>>>20))+a)|0;c+=(((d&a)|(~d&b))+k[6]-1473231341)|0;c=(((c<<17)|(c>>>15))+d)|0;b+=(((c&d)|(~c&a))+k[7]-45705983)|0;b=(((b<<22)|(b>>>10))+c)|0;a+=(((b&c)|(~b&d))+k[8]+1770035416)|0;a=(((a<<7)|(a>>>25))+b)|0;d+=(((a&b)|(~a&c))+k[9]-1958414417)|0;d=(((d<<12)|(d>>>20))+a)|0;c+=(((d&a)|(~d&b))+k[10]-42063)|0;c=(((c<<17)|(c>>>15))+d)|0;b+=(((c&d)|(~c&a))+k[11]-1990404162)|0;b=(((b<<22)|(b>>>10))+c)|0;a+=(((b&c)|(~b&d))+k[12]+1804603682)|0;a=(((a<<7)|(a>>>25))+b)|0;d+=(((a&b)|(~a&c))+k[13]-40341101)|0;d=(((d<<12)|(d>>>20))+a)|0;c+=(((d&a)|(~d&b))+k[14]-1502002290)|0;c=(((c<<17)|(c>>>15))+d)|0;b+=(((c&d)|(~c&a))+k[15]+1236535329)|0;b=(((b<<22)|(b>>>10))+c)|0;a+=(((b&d)|(c&~d))+k[1]-165796510)|0;a=(((a<<5)|(a>>>27))+b)|0;d+=(((a&c)|(b&~c))+k[6]-1069501632)|0;d=(((d<<9)|(d>>>23))+a)|0;c+=(((d&b)|(a&~b))+k[11]+643717713)|0;c=(((c<<14)|(c>>>18))+d)|0;b+=(((c&a)|(d&~a))+k[0]-373897302)|0;b=(((b<<20)|(b>>>12))+c)|0;a+=(((b&d)|(c&~d))+k[5]-701558691)|0;a=(((a<<5)|(a>>>27))+b)|0;d+=(((a&c)|(b&~c))+k[10]+38016083)|0;d=(((d<<9)|(d>>>23))+a)|0;c+=(((d&b)|(a&~b))+k[15]-660478335)|0;c=(((c<<14)|(c>>>18))+d)|0;b+=(((c&a)|(d&~a))+k[4]-405537848)|0;b=(((b<<20)|(b>>>12))+c)|0;a+=(((b&d)|(c&~d))+k[9]+568446438)|0;a=(((a<<5)|(a>>>27))+b)|0;d+=(((a&c)|(b&~c))+k[14]-1019803690)|0;d=(((d<<9)|(d>>>23))+a)|0;c+=(((d&b)|(a&~b))+k[3]-187363961)|0;c=(((c<<14)|(c>>>18))+d)|0;b+=(((c&a)|(d&~a))+k[8]+1163531501)|0;b=(((b<<20)|(b>>>12))+c)|0;a+=(((b&d)|(c&~d))+k[13]-1444681467)|0;a=(((a<<5)|(a>>>27))+b)|0;d+=(((a&c)|(b&~c))+k[2]-51403784)|0;d=(((d<<9)|(d>>>23))+a)|0;c+=(((d&b)|(a&~b))+k[7]+1735328473)|0;c=(((c<<14)|(c>>>18))+d)|0;b+=(((c&a)|(d&~a))+k[12]-1926607734)|0;b=(((b<<20)|(b>>>12))+c)|0;a+=((b^c^d)+k[5]-378558)|0;a=(((a<<4)|(a>>>28))+b)|0;d+=((a^b^c)+k[8]-2022574463)|0;d=(((d<<11)|(d>>>21))+a)|0;c+=((d^a^b)+k[11]+1839030562)|0;c=(((c<<16)|(c>>>16))+d)|0;b+=((c^d^a)+k[14]-35309556)|0;b=(((b<<23)|(b>>>9))+c)|0;a+=((b^c^d)+k[1]-1530992060)|0;a=(((a<<4)|(a>>>28))+b)|0;d+=((a^b^c)+k[4]+1272893353)|0;d=(((d<<11)|(d>>>21))+a)|0;c+=((d^a^b)+k[7]-155497632)|0;c=(((c<<16)|(c>>>16))+d)|0;b+=((c^d^a)+k[10]-1094730640)|0;b=(((b<<23)|(b>>>9))+c)|0;a+=((b^c^d)+k[13]+681279174)|0;a=(((a<<4)|(a>>>28))+b)|0;d+=((a^b^c)+k[0]-358537222)|0;d=(((d<<11)|(d>>>21))+a)|0;c+=((d^a^b)+k[3]-722521979)|0;c=(((c<<16)|(c>>>16))+d)|0;b+=((c^d^a)+k[6]+76029189)|0;b=(((b<<23)|(b>>>9))+c)|0;a+=((b^c^d)+k[9]-640364487)|0;a=(((a<<4)|(a>>>28))+b)|0;d+=((a^b^c)+k[12]-421815835)|0;d=(((d<<11)|(d>>>21))+a)|0;c+=((d^a^b)+k[15]+530742520)|0;c=(((c<<16)|(c>>>16))+d)|0;b+=((c^d^a)+k[2]-995338651)|0;b=(((b<<23)|(b>>>9))+c)|0;a+=((c^(b|~d))+k[0]-198630844)|0;a=(((a<<6)|(a>>>26))+b)|0;d+=((b^(a|~c))+k[7]+1126891415)|0;d=(((d<<10)|(d>>>22))+a)|0;c+=((a^(d|~b))+k[14]-1416354905)|0;c=(((c<<15)|(c>>>17))+d)|0;b+=((d^(c|~a))+k[5]-57434055)|0;b=(((b<<21)|(b>>>11))+c)|0;a+=((c^(b|~d))+k[12]+1700485571)|0;a=(((a<<6)|(a>>>26))+b)|0;d+=((b^(a|~c))+k[3]-1894986606)|0;d=(((d<<10)|(d>>>22))+a)|0;c+=((a^(d|~b))+k[10]-1051523)|0;c=(((c<<15)|(c>>>17))+d)|0;b+=((d^(c|~a))+k[1]-2054922799)|0;b=(((b<<21)|(b>>>11))+c)|0;a+=((c^(b|~d))+k[8]+1873313359)|0;a=(((a<<6)|(a>>>26))+b)|0;d+=((b^(a|~c))+k[15]-30611744)|0;d=(((d<<10)|(d>>>22))+a)|0;c+=((a^(d|~b))+k[6]-1560198380)|0;c=(((c<<15)|(c>>>17))+d)|0;b+=((d^(c|~a))+k[13]+1309151649)|0;b=(((b<<21)|(b>>>11))+c)|0;a+=((c^(b|~d))+k[4]-145523070)|0;a=(((a<<6)|(a>>>26))+b)|0;d+=((b^(a|~c))+k[11]-1120210379)|0;d=(((d<<10)|(d>>>22))+a)|0;c+=((a^(d|~b))+k[2]+718787259)|0;c=(((c<<15)|(c>>>17))+d)|0;b+=((d^(c|~a))+k[9]-343485551)|0;b=(((b<<21)|(b>>>11))+c)|0;x[0]=(a+x[0])|0;x[1]=(b+x[1])|0;x[2]=(c+x[2])|0;x[3]=(d+x[3])|0}function md5blk(s){var md5blks=[],i;for(i=0;i<64;i+=4){md5blks[i>>2]=s.charCodeAt(i)+(s.charCodeAt(i+1)<<8)+(s.charCodeAt(i+2)<<16)+(s.charCodeAt(i+3)<<24)}return md5blks}function md5blk_array(a){var md5blks=[],i;for(i=0;i<64;i+=4){md5blks[i>>2]=a[i]+(a[i+1]<<8)+(a[i+2]<<16)+(a[i+3]<<24)}return md5blks}function md51(s){var n=s.length,state=[1732584193,-271733879,-1732584194,271733878],i,length,tail,tmp,lo,hi;for(i=64;i<=n;i+=64){md5cycle(state,md5blk(s.substring(i-64,i)))}s=s.substring(i-64);length=s.length;tail=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];for(i=0;i<length;i+=1){tail[i>>2]|=s.charCodeAt(i)<<(i%4<<3)}tail[i>>2]|=128<<(i%4<<3);if(i>55){md5cycle(state,tail);for(i=0;i<16;i+=1){tail[i]=0}}tmp=n*8;tmp=tmp.toString(16).match(/(.*?)(.{0,8})$/);lo=parseInt(tmp[2],16);hi=parseInt(tmp[1],16)||0;tail[14]=lo;tail[15]=hi;md5cycle(state,tail);return state}function md51_array(a){var n=a.length,state=[1732584193,-271733879,-1732584194,271733878],i,length,tail,tmp,lo,hi;for(i=64;i<=n;i+=64){md5cycle(state,md5blk_array(a.subarray(i-64,i)))}a=i-64<n?a.subarray(i-64):new Uint8Array(0);length=a.length;tail=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0];for(i=0;i<length;i+=1){tail[i>>2]|=a[i]<<(i%4<<3)}tail[i>>2]|=128<<(i%4<<3);if(i>55){md5cycle(state,tail);for(i=0;i<16;i+=1){tail[i]=0}}tmp=n*8;tmp=tmp.toString(16).match(/(.*?)(.{0,8})$/);lo=parseInt(tmp[2],16);hi=parseInt(tmp[1],16)||0;tail[14]=lo;tail[15]=hi;md5cycle(state,tail);return state}function rhex(n){var s='',j;for(j=0;j<4;j+=1){s+=hex_chr[(n>>(j*8+4))&15]+hex_chr[(n>>(j*8))&15]}return s}function hex(x){var i;for(i=0;i<x.length;i+=1){x[i]=rhex(x[i])}return x.join('')}if(hex(md51('hello'))!=='5d41402abc4b2a76b9719d911017c592'){add32=function(x,y){var lsw=(x&65535)+(y&65535),msw=(x>>16)+(y>>16)+(lsw>>16);return(msw<<16)|(lsw&65535)}}if(typeof ArrayBuffer!=='undefined'&&!ArrayBuffer.prototype.slice){(function(){function clamp(val,length){val=val|0||0;if(val<0){return Math.max(val+length,0)}return Math.min(val,length)}ArrayBuffer.prototype.slice=function(from,to){var length=this.byteLength,begin=clamp(from,length),end=length,num,target,targetArray,sourceArray;if(to!==undefined){end=clamp(to,length)}if(begin>end){return new ArrayBuffer(0)}num=end-begin;target=new ArrayBuffer(num);targetArray=new Uint8Array(target);sourceArray=new Uint8Array(this,begin,num);targetArray.set(sourceArray);return target}})()}function toUtf8(str){if(/[\u0080-\uFFFF]/.test(str)){str=unescape(encodeURIComponent(str))}return str}function utf8Str2ArrayBuffer(str,returnUInt8Array){var length=str.length,buff=new ArrayBuffer(length),arr=new Uint8Array(buff),i;for(i=0;i<length;i+=1){arr[i]=str.charCodeAt(i)}return returnUInt8Array?arr:buff}function arrayBuffer2Utf8Str(buff){return String.fromCharCode.apply(null,new Uint8Array(buff))}function concatenateArrayBuffers(first,second,returnUInt8Array){var result=new Uint8Array(first.byteLength+second.byteLength);result.set(new Uint8Array(first));result.set(new Uint8Array(second),first.byteLength);return returnUInt8Array?result:result.buffer}function hexToBinaryString(hex){var bytes=[],length=hex.length,x;for(x=0;x<length-1;x+=2){bytes.push(parseInt(hex.substr(x,2),16))}return String.fromCharCode.apply(String,bytes)}function SparkMD5(){this.reset()}SparkMD5.prototype.append=function(str){this.appendBinary(toUtf8(str));return this};SparkMD5.prototype.appendBinary=function(contents){this._buff+=contents;this._length+=contents.length;var length=this._buff.length,i;for(i=64;i<=length;i+=64){md5cycle(this._hash,md5blk(this._buff.substring(i-64,i)))}this._buff=this._buff.substring(i-64);return this};SparkMD5.prototype.end=function(raw){var buff=this._buff,length=buff.length,i,tail=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],ret;for(i=0;i<length;i+=1){tail[i>>2]|=buff.charCodeAt(i)<<(i%4<<3)}this._finish(tail,length);ret=hex(this._hash);if(raw){ret=hexToBinaryString(ret)}this.reset();return ret};SparkMD5.prototype.reset=function(){this._buff='';this._length=0;this._hash=[1732584193,-271733879,-1732584194,271733878];return this};SparkMD5.prototype.getState=function(){return{buff:this._buff,length:this._length,hash:this._hash.slice()}};SparkMD5.prototype.setState=function(state){this._buff=state.buff;this._length=state.length;this._hash=state.hash;return this};SparkMD5.prototype.destroy=function(){delete this._hash;delete this._buff;delete this._length};SparkMD5.prototype._finish=function(tail,length){var i=length,tmp,lo,hi;tail[i>>2]|=128<<(i%4<<3);if(i>55){md5cycle(this._hash,tail);for(i=0;i<16;i+=1){tail[i]=0}}tmp=this._length*8;tmp=tmp.toString(16).match(/(.*?)(.{0,8})$/);lo=parseInt(tmp[2],16);hi=parseInt(tmp[1],16)||0;tail[14]=lo;tail[15]=hi;md5cycle(this._hash,tail)};SparkMD5.hash=function(str,raw){return SparkMD5.hashBinary(toUtf8(str),raw)};SparkMD5.hashBinary=function(content,raw){var hash=md51(content),ret=hex(hash);return raw?hexToBinaryString(ret):ret};SparkMD5.ArrayBuffer=function(){this.reset()};SparkMD5.ArrayBuffer.prototype.append=function(arr){var buff=concatenateArrayBuffers(this._buff.buffer,arr,true),length=buff.length,i;this._length+=arr.byteLength;for(i=64;i<=length;i+=64){md5cycle(this._hash,md5blk_array(buff.subarray(i-64,i)))}this._buff=i-64<length?new Uint8Array(buff.buffer.slice(i-64)):new Uint8Array(0);return this};SparkMD5.ArrayBuffer.prototype.end=function(raw){var buff=this._buff,length=buff.length,tail=[0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],i,ret;for(i=0;i<length;i+=1){tail[i>>2]|=buff[i]<<(i%4<<3)}this._finish(tail,length);ret=hex(this._hash);if(raw){ret=hexToBinaryString(ret)}this.reset();return ret};SparkMD5.ArrayBuffer.prototype.reset=function(){this._buff=new Uint8Array(0);this._length=0;this._hash=[1732584193,-271733879,-1732584194,271733878];return this};SparkMD5.ArrayBuffer.prototype.getState=function(){var state=SparkMD5.prototype.getState.call(this);state.buff=arrayBuffer2Utf8Str(state.buff);return state};SparkMD5.ArrayBuffer.prototype.setState=function(state){state.buff=utf8Str2ArrayBuffer(state.buff,true);return SparkMD5.prototype.setState.call(this,state)};SparkMD5.ArrayBuffer.prototype.destroy=SparkMD5.prototype.destroy;SparkMD5.ArrayBuffer.prototype._finish=SparkMD5.prototype._finish;SparkMD5.ArrayBuffer.hash=function(arr,raw){var hash=md51_array(new Uint8Array(arr)),ret=hex(hash);return raw?hexToBinaryString(ret):ret};return SparkMD5}export default factory();

标签:function,const,js,15,webWorker,file,分片,new,多线程
From: https://blog.csdn.net/m0_56344834/article/details/144168131

相关文章

  • 【探讨】批量操作以及多线程下保证事务的一致性
    1 前言假如给你一个场景,有一批1万或者10万的数据,让你插入到数据库中怎么做呢?我们这节来看看。首先一点我们单纯的一个个INSERT语句,我们就不试了,这一个个的肯定慢,我们这里统一用INSERTINTO表(字段1,字段2)VALUES(值1,值2),(值11,值22),(值111,值222);这种方式分批跑高效点。......
  • Java多线程介绍及使用指南
    “多线程”:并发要介绍线程,首先要区分开程序、进程和线程这三者的区别。程序:具有一定功能的代码的集合,但是是静态的,没有启动运行进程:启动运行的程序【资源的分配单位】线程:进程中的每一条执行路径,就是线程。概念:并行:多个CPU同时执行多个任务并发:一个CPU“同时”执行多......
  • Java面试之多线程&并发篇(9)
    前言本来想着给自己放松一下,刷刷博客,突然被几道面试题难倒!引用类型有哪些?有什么区别?说说你对JMM内存模型的理解?为什么需要JMM?多线程有什么用?似乎有点模糊了,那就大概看一下面试题吧。好记性不如烂键盘***12万字的java面试题整理******java核心面试知识整理******Java高频......
  • JAVA之多线程
    什么是线程?线程(Thread)是一个程序内部的一条执行流程。什么是多线程?多线程是指从软硬件上实现的多条执行流程的技术(多条线程由CPU负责调度执行)如何创建多线程:有两种方法可以创建新的执行线程。一种是将类声明为Thread的子类。此子类应覆盖类Thread的run方法。然后可......
  • java实现多线程读,单线程写
    多线程读,单线程写excel通过工厂模式,队列实现多线程读取(可以是文件,也可以是数据库),单线程向excel文件写;实现了一个线程出现问题其他线程同步中断。该实现逻辑的同时可以防止oom。  privatestaticfinalintINITIAL_CAPACITY=10;  privatestaticfinalintMAX_CA......
  • java 多线程同步方法CyclicBarrier/CountDownLatch/AtomicBoolean/Semaphore
    CyclicBarrier 有两个构造函数,CyclicBarrier(int),CyclicBarrier(int,Runnable)目的:通过输入任务数实现线程同步; 使用场景: 多线程计算:当多个线程需要分阶段并行处理数据,但在每一阶段结束时需要所有线程同步,以便开始下一阶段的处理。 并行任务协调:例如,在并行搜索或并行数......
  • 多线程判断redis key导致Java hep space内存溢出
     线上经常发现报如下错误,后来发现rootcause是selectConfigitemforinstanceMap这个方法会查出几十万的结果集然后json压缩解压,写map等等操作,但是这个selectConfigitemforinstanceMap方法被很多地方调用到了,导致极有可能多线程同时都在查出几十万的结果集然后json压缩解压,写map等......
  • MongoDB测试环境搭建分片脚本
    搭建1个config节点,一个mongos节点,两个分片,每个分片3个节点。传入参数为mongos节点端口号,config节点端口号,分片节点端口号可根据需求修改存放的目录basedir#!/bin/bash#定义usage函数usage(){echo"Usage:mongos_portconfig_portshard_node_port_1....shard_......
  • java 多线程传统锁:synchronized,Lock锁
    传统锁:synchronized是内置关键词,无法获取锁的状态Lock锁:类是否获取到锁,需手动释放锁publicclassSaleTicketDem{publicstaticvoidmain(String[]args){Ticket2t=newTicket2();newThread(()->{for(inti=0;i<40;i++){t.sale();;}},"a").start();n......
  • 【Linux】多线程(POSIX信号量、线程池、线程安全)
    ......