var data = { Array.from(Array(10000).keys()) };
Object.keys(data)
.filter(key => Number(key) % 2 === 0)
.reduce((acc, key) => {
acc[key] = data[key];
return acc;
}, {});
Object.keys(data)
.filter(key => Number(key) % 2 === 0)
.reduce((acc, key) => {
acc.set(key, data[key]);
return acc;
}, new Map());
const filter = (val) => {
const result = {}
for (const key in val) {
if (Number(key) % 2 === 0) {
continue;
}
result[key] = val[key];
}
return result;
};
const filter = (val) => {
const result = new Map();
for (const key in val) {
if (Number(key) % 2 === 0) {
continue;
}
result.set(key, val[key]);
}
return Object.fromEntries(result.entries());
};
--enable-precise-memory-info
flag.
Test case name | Result |
---|---|
Reduce (object) | |
Reduce (Map) | |
for-of (object) | |
for-of (Map) |
Test name | Executions per second |
---|---|
Reduce (object) | 2439.2 Ops/sec |
Reduce (Map) | 1638.2 Ops/sec |
for-of (object) | 1851825536.0 Ops/sec |
for-of (Map) | 1862645120.0 Ops/sec |
The benchmark on MeasureThat.net titled "Object filtering by key" is designed to compare different methods for filtering keys in a JavaScript object. It specifically tests various approaches for retrieving and constructing a filtered result based on whether the key (converted to a number) is even or odd.
Reduce with Object: Reduce (object)
Object.keys(data)
.filter(key => Number(key) % 2 === 0)
.reduce((acc, key) => {
acc[key] = data[key];
return acc;
}, {});
Object.keys()
to extract keys from the object, filter()
to find even keys, and reduce()
to construct a new object.Reduce with Map: Reduce (Map)
Object.keys(data)
.filter(key => Number(key) % 2 === 0)
.reduce((acc, key) => {
acc.set(key, data[key]);
return acc;
}, new Map());
Map
instead of an object. Maps preserve the insertion order of keys, which can be beneficial.For-of Loop with Object: for-of (object)
const filter = (val) => {
const result = {};
for (const key in val) {
if (Number(key) % 2 === 0) {
continue;
}
result[key] = val[key];
}
return result;
};
for...in
loop directly to iterate through the object's keys to filter even ones and build a new object without additional data structures.For-of Loop with Map: for-of (Map)
const filter = (val) => {
const result = new Map();
for (const key in val) {
if (Number(key) % 2 === 0) {
continue;
}
result.set(key, val[key]);
}
return Object.fromEntries(result.entries());
};
for...in
loop but constructs a Map
and then converts it back to an object at the end.Reduce Methods (Object & Map):
For-of Loop Methods (Object & Map):
Performance Data: Based on the benchmark results, the for-of (Map)
performed the best, followed closely by for-of (object)
. The two Reduce
methods were significantly slower, indicating that functional chaining can incur performance penalties due to the overhead of managing multiple intermediate steps.
Use of Maps: Maps are particularly useful in scenarios where the order of keys matters or when keys are not limited to strings. They also have better performance characteristics when it comes to frequent additions and removals of key-value pairs.
Other alternatives that may not have been tested include:
Using libraries such as Lodash, which provides utility functions for object and array manipulations. For example, _.pickBy()
allows for more concise filtering based on predicate functions.
Native JavaScript features like the spread operator or Object.entries() could also be leveraged to achieve object filtering in a more modern or functional style.
In conclusion, this benchmark provides insights into the performance and usability of different methods for filtering objects in JavaScript, highlighting the trade-offs between performance, readability, and the complexity of different coding paradigms.