如何将一个NodeJS二进制缓冲区转换为JavaScript数组缓冲区?


当前回答

使用以下优秀的npm包:to-arraybuffer。

或者,您可以自己实现它。如果你的缓冲区叫buf,这样做:

buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength)

其他回答

现在有一个非常有用的npm包:buffer https://github.com/feross/buffer

它试图提供一个API,是100%相同的节点的缓冲区API,并允许:

将类型化数组转换为缓冲区:https://github.com/feross/buffer#convert-typed-array-to-buffer 将缓冲区转换为类型化数组:https://github.com/feross/buffer#convert-buffer-to-typed-array

还有更多。

没有依赖,最快,Node.js 4。X及以后

Buffers是Uint8Arrays,所以你只需要切片(复制)它的支持ArrayBuffer的区域。

// Original Buffer
let b = Buffer.alloc(512);
// Slice (copy) its segment of the underlying ArrayBuffer
let ab = b.buffer.slice(b.byteOffset, b.byteOffset + b.byteLength);

切片和偏移量是必需的,因为小缓冲区(默认小于4 kB,池大小的一半)可以是共享ArrayBuffer上的视图。如果没有切片,你可能会得到一个ArrayBuffer,其中包含了来自另一个Buffer的数据。请参阅文档中的解释。

如果你最终需要一个TypedArray,你可以创建一个而不复制数据:

// Create a new view of the ArrayBuffer without copying
let ui32 = new Uint32Array(b.buffer, b.byteOffset, b.byteLength / Uint32Array.BYTES_PER_ELEMENT);

没有依赖,速度适中,适用于任何版本的Node.js

使用Martin Thomson的答案,它在O(n)时间内运行。(参见我对他关于非优化的回答的回复。使用DataView很慢。即使你需要转换字节,也有更快的方法。)

依赖,快速,Node.js≤0.12或iojs 3.x

您可以使用https://www.npmjs.com/package/memcpy向任何一个方向(缓冲区到ArrayBuffer和返回)。它比这里发布的其他答案要快,而且是一个写得很好的库。节点0.12到iojs 3。X要求ngossen's fork(见此)。

使用以下优秀的npm包:to-arraybuffer。

或者,您可以自己实现它。如果你的缓冲区叫buf,这样做:

buf.buffer.slice(buf.byteOffset, buf.byteOffset + buf.byteLength)

可以把ArrayBuffer看作是类型化的Buffer。

因此,ArrayBuffer总是需要一个类型(所谓的“数组缓冲区视图”)。通常,数组缓冲区视图有Uint8Array或Uint16Array类型。

Renato Mangini有一篇关于ArrayBuffer和String之间转换的好文章。

我在一个代码示例中总结了基本部分(用于Node.js)。它还展示了如何在类型化的ArrayBuffer和非类型化的Buffer之间进行转换。

function stringToArrayBuffer(string) {
  const arrayBuffer = new ArrayBuffer(string.length);
  const arrayBufferView = new Uint8Array(arrayBuffer);
  for (let i = 0; i < string.length; i++) {
    arrayBufferView[i] = string.charCodeAt(i);
  }
  return arrayBuffer;
}

function arrayBufferToString(buffer) {
  return String.fromCharCode.apply(null, new Uint8Array(buffer));
}

const helloWorld = stringToArrayBuffer('Hello, World!'); // "ArrayBuffer" (Uint8Array)
const encodedString = new Buffer(helloWorld).toString('base64'); // "string"
const decodedBuffer = Buffer.from(encodedString, 'base64'); // "Buffer"
const decodedArrayBuffer = new Uint8Array(decodedBuffer).buffer; // "ArrayBuffer" (Uint8Array)

console.log(arrayBufferToString(decodedArrayBuffer)); // prints "Hello, World!"

令人惊讶的是,在我的情况下(电子:发送缓冲区渲染器ArrayBuffer),这只是工作

function bufferToArrayBuffer(buffer: Buffer): ArrayBuffer {
  return buffer.buffer as ArrayBuffer;
}

快了100倍

function bufferToArrayBuffer(buf) {
    const ab = new ArrayBuffer(buf.length);
    const view = new Uint8Array(ab);
    for (let i = 0; i < buf.length; ++i) {
        view[i] = buf[i];
    }
    return ab;
}