初识web Worker(二)

316 阅读1分钟

以前经常听说过web worker,但是见过没有过真正的使用场景。然后目前要做一个项目在讨论中发现可以用到web worker,刚好研究一下web worker是如何使用的。

书接上文初识web Worker(一)

3.基本用法

  • worker线程
// worker线程内部监听message事件,self代表worker线程自身,即子线程的全局对象
// 也可以使用`self.onmessage`指定。监听函数的参数是一个事件对象,它的data属性包含主线程发来的数据。
// self.postMessage()方法用来向主线程发送消息。
self.addEventListener('message', function (e) {
  self.postMessage('You said: ' + e.data);
}, false);
self.onmessage = function(e){
  self.postMessage('You said: ' + e.data);
}

// 使用完毕,为了节省系统资源,必须关闭worker
self.close()

4.数据通信

主线程与worker线程之间的通信内容可以是:文本、对象、二进制数据。

// 主线程
var uInt8Array = new Uint8Array(new ArrayBuffer(10));
for (var i = 0; i < uInt8Array.length; ++i) {
  uInt8Array[i] = i * 2; // [0, 2, 4, 6, 8,...]
}
worker.postMessage(uInt8Array);

// Worker 线程
self.onmessage = function (e) {
  var uInt8Array = e.data;
  postMessage('Inside worker.js: uInt8Array.toString() = ' + uInt8Array.toString());
  postMessage('Inside worker.js: uInt8Array.byteLength = ' + uInt8Array.byteLength);
};

5.实例

// worker线程
self.onmessage = function(e){
    transAudioData.transcode(e.data)
}
let transAudioData = {
    transcode(audioData) {
      let output = transAudioData.to16kHz(audioData)
      output = transAudioData.to16BitPCM(output)
      output = Array.from(new Uint8Array(output.buffer))
      self.postMessage(output)
      // return output
    },
    to16kHz(audioData) {
      var data = new Float32Array(audioData)
      var fitCount = Math.round(data.length * (16000 / 44100))
      var newData = new Float32Array(fitCount)
      var springFactor = (data.length - 1) / (fitCount - 1)
      newData[0] = data[0]
      for (let i = 1; i < fitCount - 1; i++) {
        var tmp = i * springFactor
        var before = Math.floor(tmp).toFixed()
        var after = Math.ceil(tmp).toFixed()
        var atPoint = tmp - before
        newData[i] = data[before] + (data[after] - data[before]) * atPoint
      }
      newData[fitCount - 1] = data[data.length - 1]
      return newData
    },
    to16BitPCM(input) {
      var dataLength = input.length * (16 / 8)
      var dataBuffer = new ArrayBuffer(dataLength)
      var dataView = new DataView(dataBuffer)
      var offset = 0
      for (var i = 0; i < input.length; i++, offset += 2) {
        var s = Math.max(-1, Math.min(1, input[i]))
        dataView.setInt16(offset, s < 0 ? s * 0x8000 : s * 0x7fff, true)
      }
      return dataView
    },
}