如何处理对同一Web Worker的多个请求

时间:2016-12-03 10:51:19

标签: javascript html5 web-worker

我创建了一个处理大数据的Web worker。以下是代码实现:

worker.js

self.addEventListener('message', function(e) {
  console.log('Message received from main script');
  for(var i=0; i<999999999; i++){
    // computation here
  }
  console.log('Posting message back to main script');
  self.postMessage('You said: ' + e.data);
}, false);

主脚本

var myWorker;
if (window.Worker) {
  if(typeof(w)=="undefined"){
    myWorker = new Worker("scripts/controllers/workers.js");
  }
}
myWorker.addEventListener('message', function(e) {
    console.log('Message received from worker'+ e.data);
}, false);

function callWorker(someData){
  myWorker.postMessage(someData);
}
callWorker(someData);//first time call - by user click
callWorker(someOtherData);//second time call - again user click (don't want to disable button)

在这种情况下,我不想多次执行callWorker。 我想要的是,如果callWorker当前正在执行并且意味着第二个请求加注callWorker,则第一个请求应该终止,第二个请求应该处理(最新请求应该处理)。

我怎样才能做到这一点?谢谢你的帮助

3 个答案:

答案 0 :(得分:1)

我猜你可以尝试终止工作人员并重新启动它。 当您调用terminate方法时,它将立即终止,而无需等待操作完成。

myWorker.terminate();
myWorker = new Worker('worker.js');

答案 1 :(得分:1)

  

如果callWorker当前正在执行并且意味着第二个请求为callWorker引发,则第一个请求应该终止并且第二个请求应该处理(最新请求应该处理)。

我担心如果不终止工作人员,你就无法做到这一点。我认为您应该等到完成后再发送下一条消息,而不是尝试终止当前进程。

答案 2 :(得分:0)

我在这里参加派对有点晚了,但我只想分享我的解决方案:

我有同样的问题,在我的情况下,等待将花费太长时间并且创建新工作人员的效率不够高。

我的解决方案是:

  1. 以增量方式在worker中执行任务。
  2. 回到主帖并“询问”是否应该继续。
  3. 主线程以一个标志响应,指示我们的工作人员是否应该继续当前任务,或者是否应该使用新数据中断/重新开始。
  4. 工人代码:

    self.addEventListener("message", function (e) {
    
        var args = new Int16Array(e.data);
    
        if (args[0] === 1) {//Start new task/start over. 
    
            ExecuteTask(args);
            self.postMessage("CONTINUE?");
    
        }else if (args[0] === 2) {//Continue 
    
            ExecuteTask(args);
            self.postMessage("CONTINUE?");
        }else{
           self.postMessage("TASK_TERMINATED");
        }
    }
    

    主线程:

    myWorker.addEventListener('message', function (e) {
        if (e.data === "CONTINUE?") {
    
            // ask the worker to start over with new data
            if (newTask) { 
                myWorker.postMessage(newDataMessage);
            }
            //No new tasks. Let the worker know it can continue. 
            else{
                myWorker.postMessage(continueMessage);
            }
        } else if (e.data === "TASK_EXECUTED" || e.data === "TASK_TERMINATED"  ) {
            workerIdle = true; //let main thread know it can execute the next new task instead of queueing it up.
        }                   
    });