<script src="https://cdn.jsdelivr.net/npm/lodash@4.17.4/lodash.min.js"></script>
<script>
function deepEquals(x, y, exceptionName) {
if (x === y) {
return true; // if both x and y are null or undefined and exactly the same
}
else if (!(x instanceof Object) || !(y instanceof Object)) {
return false; // if they are not strictly equal, they both need to be Objects
}
else if (x.constructor !== y.constructor) {
// they must have the exact same prototype chain, the closest we can do is
// test their constructor.
return false;
}
else {
for (const p in x) {
if (!x.hasOwnProperty(p)) {
continue; // other properties were tested using x.constructor === y.constructor
}
if (!y.hasOwnProperty(p)) {
return false; // allows to compare x[ p ] and y[ p ] when set to undefined
}
if (x[p] === y[p] || (!!exceptionName && p === exceptionName)) {
continue; // if they have the same strict value or identity then they are equal
}
if (typeof x[p] !== 'object') {
return false; // Numbers, Strings, Functions, Booleans must be strictly equal
}
if (!deepEquals(x[p], y[p])) {
return false;
}
}
for (const p in y) {
if (y.hasOwnProperty(p) && !x.hasOwnProperty(p)) {
return false;
}
}
return true;
}
}
</script>
// 1 level deep
window.foo1 = { a: 1, b: 2, c: { a: 1, b: 2, c: { a: 1, b: 2 } } };
window.bar1 = { a: 1, b: 3, c: { a: 1, b: 2, c: { a: 1, b: 2 } } };
// 2 levels deep
window.foo2 = { a: 1, b: 2, c: { a: 1, b: 2, c: { a: 1, b: 2 } } };
window.bar2 = { a: 1, b: 2, c: { a: 1, b: 3, c: { a: 1, b: 2 } } };
// 3 levels deep
window.foo3 = { a: 1, b: 2, c: { a: 1, b: 2, c: { a: 1, b: 2 } } };
window.bar3 = { a: 1, b: 2, c: { a: 1, b: 2, c: { a: 1, b: 4 } } };
_.isEqual(window.foo1, window.bar1)
deepEquals(window.foo1, window.bar1);
_.isEqual(window.foo2, window.bar2)
deepEquals(window.foo2, window.bar2);
_.isEqual(window.foo3, window.bar3)
deepEquals(window.foo3, window.bar3);
window.foo3 === window.bar3
--enable-precise-memory-info
flag.
Test case name | Result |
---|---|
_.isEqual Level 1 | |
deepEquals Level 1 | |
_.isEqual Level 2 | |
deepEquals Level 2 | |
_.isEqual Level 3 | |
deepEquals Level 3 | |
native equals |
Test name | Executions per second |
---|---|
_.isEqual Level 1 | 1275986.4 Ops/sec |
deepEquals Level 1 | 3346476.5 Ops/sec |
_.isEqual Level 2 | 829500.2 Ops/sec |
deepEquals Level 2 | 2742466.5 Ops/sec |
_.isEqual Level 3 | 621118.8 Ops/sec |
deepEquals Level 3 | 2346973.2 Ops/sec |
native equals | 5631591.0 Ops/sec |
I'll break down the provided benchmark test cases and explain what's being tested, compared, and their pros/cons.
Benchmark Test Cases:
The benchmark test cases are designed to compare the performance of three different implementations:
_.isEqual
(Lodash library)deepEquals
function===
)Each test case measures the execution time for each implementation, allowing users to compare their performance.
Test Case Breakdown:
There are six test cases in total, with three levels of depth:
_.isEqual(window.foo1, window.bar1)
and deepEquals(window.foo1, window.bar1)
These tests compare two objects with similar properties._.isEqual(window.foo2, window.bar2)
and deepEquals(window.foo2, window.bar2)
These tests compare two objects with more complex structures._.isEqual(window.foo3, window.bar3)
and deepEquals(window.foo3, window.bar3)
These tests compare two deeply nested objects.Comparison:
The benchmark results show the execution time for each implementation:
===
):_.isEqual
(Lodash library):deepEquals
function:Pros/Cons:
Native Equality Operator (===
):
Pros:
Cons:
Lodash Library (_.isEqual
):
Pros:
Cons:
Custom deepEquals
Function:
Pros:
Cons: