Binary data in JavaScript comes in a few forms:
To work with this binary data we have a few options:
Bitwise Operations
Bitwise operations allow us to change bits of numbers and create new numbers from two inputs. They are useful for encoding and decoding data.
const bigInt = (2n ** 64n) >> 32n; console.log(bigInt); // 4294967296n const normalNumber = (2 ** 64) >> 32; console.log(normalNumber); // 0
- AND
The & operation sets a bit to 1 if only the bit it is being compared to is also 1.
const v1 = 0b1111; const v2 = 0b1011; const result = v1 & v2; // 0b1011
- OR
The | operation sets a bit to 1 if one of the bits being compared is 1.
const v1 = 0b1111; const v2 = 0b1011; const result = v1 | v2; // 0b1111
- XOR
The ^ operation sets a bit to 1 if only one of the bits being compared is 1 otherwise it sets it to 0.
const v1 = 0b1111; const v2 = 0b1011; const result = v1 ^ v2; // 0b0100
- NOT
The ~ operation flips all bits.
const result = ~0b1011; // 0b0100
- Zero Fill Left Shift
The << operation shifts bits to the left by a given amount.
const result = 0b0011 << 2; // 0b1100
- Signed Right Shift
The >>> operation shifts bits to the right by a given amount and preserves the sign if the number is negative.
const result = -5 >> 2; // -2
This operation can also be used for flooring a number:
const result = 1.2 >> 0; // 1
- Zero Fill Right Shift
The >>> operation shifts bits to the right by a given amount.
const result1 = -5 >>> 2; // 1073741822 const result2 = 0b1100 >>> 2; // 0b0011
- Encoding And Decoding Data
Here is a basic example showing how you could use a single number to store 8 numbers with a range of 0 to 15.
const mask = 0xf; const bitSize = 4; const getValue = (data: number, index: number) => { index *= bitSize; return ((mask << index) & data) >>> index; } const setValue = (data: number, index: number, value: number) => { index *= bitSize; return (data & ~(mask << index)) | ((value & mask) << index); }
So, we have two functions here. The getValue function will take the encoded number and return the value at the given index. While the setValue function will take the encoded number and encode the provided value at the given index and then return the new encoded number.
Buffers
JavaScript uses buffers or "byte arrays" to work with raw binary data. You can not directly access the data. You need to use another object such as a Typed Array to work with the bytes of the buffer.
Buffers are useful because they can be sent to and shared with other threads without copying the data. They can also be easily sent and received through web sockets, compressed, and saved to the file system.
- ArrayBuffer
An ArrayBuffer object is a fixed length array of bytes. You can make one like this:
const buffer = new ArrayBuffer(4);
In this case we set the length of the buffer to be 4 bytes. So, if we created a TypedArray that used 1 byte per number we could store four numbers in that buffer.
- SharedArrayBuffer
A SharedArrayBuffer is basically the same as the ArrayBuffer but it can be shared by multiple threads.
const buffer = new SharedArrayBuffer(4);
When an ArrayBuffer is transferred to another thread the thread that sent it losses access to it. The SharedArrayBuffer then basically acts as a pointer to the same buffer.
Typed Arrays
Typed Arrays are fixed length arrays that act as a view over a buffer. They can either be created with or without a buffer.
Here are the typed arrays:
// 1 byte per number | range: -128 - 127 const int8 = new Int8Array(); // 1 byte per number | range: 0 - 255 const uInt8 = new Uint8Array(); // 1 byte per number | range: 0 - 255 const clampedInt8 = new Uint8ClampedArray(); // 2 bytes per number | range: -32768 - 32767 const int16 = new Int16Array(); // 2 bytes per number | range: 0 - 65535 const uInt16 = new Uint16Array(); // 4 bytes per number | range: -2147483648 - 2147483647 const int32 = new Int32Array(); // 4 bytes per number | range: 0 - 4294967295 const uInt32 = new Uint32Array(); // 4 bytes per number | range: -3.4E38 - 3.4E38 const float32 = new Float32Array(); // 4 bytes per number | range: -1.8E308 - 1.8E308 const float64 = new Float64Array(); // 4 bytes per number | range: -2^63 - 2^63 - 1 const int65 = new BigInt64Array(); // 4 bytes per number | range: 0 - 2^64 - 1 const uInt65 = new BigUint64Array();
You can create a typed array in a few ways:
// Creates a typed array with the given length const ex1 = new Uint16Array(4); // Creates a typed array from an array const ex2 = new Uint16Array([1,2,3,4]); // Creates a typed array from an ArrayBuffer const byteSize = 4 * 2; const buffer = new ArrayBuffer(byteSize); const ex3 = new Uint16Array(buffer); // Creates a typed array from an ArrayBuffer at a given index and byte length const buffer2 = new ArrayBuffer(byteSize * 2); const ex4 = new Uint16Array(buffer, byteSize, byteSize); // Creates a typed array from a SharedArrayBuffer const SAB = new SharedArrayBuffer(byteSize) const ex5 = new Uint16Array(SAB);
You can use a Typed Array by indexing it like a normal array:
const data = new Uint16Array(4); data[0] = 300; data[1] = 1000; data[2] = 10_000; data[3] = 5_000;
DataViews
The DataView object lets you set and get sets of bytes from buffers.
Here are its functions:
const buffer = new ArrayBuffer(16); const dv = new DataView(buffer); // 1 byte signed int const int8 = dv.getInt8(0); dv.setInt8(0,int8); // 1 byte un-signed int const uInt8 = dv.getUint8(0); dv.setUint8(0,uInt8); // 2 byte signed int const int16 = dv.getInt16(0); dv.setInt16(0,int16); // 2 byte un-signed int const uInt16 = dv.getUint16(0); dv.setUint16(0,uInt16); // 4 byte signed int const int32 = dv.getInt32(0); dv.setInt32(0,int32); // 4 byte un-signed int const uInt32 = dv.getUint32(0); dv.setUint32(0,uInt32); // 4 byte float const float32 = dv.getFloat32(0); dv.setFloat32(0,float32); // 8 byte float const float64 = dv.getFloat64(0); dv.setFloat64(0,float64); // 8 byte signed big int const int64 = dv.getBigInt64(0); dv.setBigInt64(0,int64); // 8 byte un-signed big int const uInt64 = dv.getBigUint64(0); dv.setBigUint64(0,uInt64);
You can create a DataView object in a few ways:
const buffer = new ArrayBuffer(10); // Create a DataView from an ArrayBuffer const dv = new DataView(buffer); // Create a DataView from part an ArrayBuffer with a given offset and length const dv2 = new DataView(buffer,5,5); // Create a DataView from a SharedArrayBuffer const SAB = new SharedArrayBuffer(10); const dv3 = new DataView(SAB);
Here is a basic example of using a 3 byte ArrayBuffer and a DataView object to update different parts of the buffer.
const buffer = new ArrayBuffer(3); const dv = new DataView(buffer); dv.setUint8(0, 10); dv.setUint16(1, 300); // value is 10 dv.getUint8(0); // value is 300 dv.getUint16(1);