<span></span>
let span = document.body.querySelector("span");
class Float32ArrayExtended extends Float32Array {
constructor (val) {
super(1);
this[0] = val;
}
get x () { return this[0]; }
set x (v) { this[0] = v; }
}
class Int32ArrayExtended extends Int32Array {
constructor (val) {
super(1);
this[0] = val;
}
get x () { return this[0]; }let
set x (v) { this[0] = v; }
}
const size = 100000;
const array = new Array(16).fill(1);
const arrayEF32 = new Float32ArrayExtended (16);
const arrayF32 = new Float32Array (16);
const arrayEI32 = new Int32ArrayExtended (16);
const arrayI32 = new Int32Array (16);
let x = 1;
const obj = {
get x() {
return array[0];
},
set x(value) {
array[0] = value;
}
};
let sum = 0;
sum = 0;
for (let i = 0; i < size; i++) {
array[0] = Math.random () * 1000;
sum += array[0];
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
obj.x = Math.random () * 1000;
sum += obj.x;
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
arrayF32[0] = Math.random () * 1000;
sum += arrayF32[0];
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
arrayEF32.x = Math.random () * 1000;
sum += arrayEF32.x;
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
arrayI32[0] = Math.random () * 1000;
sum += arrayI32[0];
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
arrayEI32.x = Math.random () * 1000;
sum += arrayEI32.x;
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
x = Math.random () * 1000;
sum += x;
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
array[0] = Math.random () * 1000;
sum += array[0];
}
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++)
sum += Math.random () * 1000;
span.innerText = sum;
sum = 0;
for (let i = 0; i < size; i++) {
const x = Math.random () * 1000;
sum += x;
}
span.innerText = sum;
--enable-precise-memory-info
flag.
Test case name | Result |
---|---|
array[0] | |
set / get | |
float 32 | |
extended float 32 | |
int 32 | |
extended int 32 | |
simple | |
array[0] 2 | |
simple 2 | |
simple 3 |
Test name | Executions per second |
---|---|
array[0] | 68.9 Ops/sec |
set / get | 32.5 Ops/sec |
float 32 | 63.0 Ops/sec |
extended float 32 | 70.8 Ops/sec |
int 32 | 72.5 Ops/sec |
extended int 32 | 77.8 Ops/sec |
simple | 40.5 Ops/sec |
array[0] 2 | 75.7 Ops/sec |
simple 2 | 77.2 Ops/sec |
simple 3 | 79.2 Ops/sec |
This benchmark compares the performance of various methods for accessing and modifying data in different array-like structures in JavaScript, specifically focusing on:
Array
)Float32Array
and Int32Array
)Simple Assignment (simple
, simple 2
, simple 3
):
x
) and summing values in a loop.Array Access (array[0]
, array[0] 2
):
Float32Array:
Float32ArrayExtended:
Float32Array
that adds custom getters and setters for handling the first element.Int32Array:
Float32Array
, this is used for handling 32-bit integer values and is benchmarked for access and mutation.Float32Array
, it is restrictive in terms of type flexibility.Int32ArrayExtended:
Int32Array
with custom getter and setter methods.Float32ArrayExtended
, enhancing usability with property syntax.Object with Getters/Setters (set / get
):
x
to access and modify an internal array.The benchmark indicates the trade-offs between performance and usability. Direct assignments to variables or accessed through simple arrays yield better performance but may sacrifice readability or flexibility. In contrast, extended classes that use getters/setters provide clear, maintainable code, but at the cost of speed because of the function overhead.
ArrayBuffer
can be preferred, often paired with typed arrays.In conclusion, this benchmark serves as an excellent foundation for understanding the performance implications of using different types of data access patterns and structures in JavaScript, particularly for scenarios that require high-performance numeric computations. Each approach has its benefits and drawbacks, and the best choice highly depends on the specific requirements of the application.