2628. JSON Deep Equal
Problem Description
The problem requires developing a function to check if two values, o1
and o2
, are deeply equal. This concept of deep equality extends beyond surface-level comparison (e.g., simply checking if two numbers are the same) and delves into structured data such as arrays and objects. Here, the comparison must be thorough:
- For primitive values (like numbers or strings), they are deeply equal if they are strictly equal (
===
). - For arrays, they are deeply equal if they have the same number of elements, each element is in the same order, and each corresponding pair of elements is also deeply equal.
- For objects (ignoring those that are instances of classes), they are deeply equal if they have the same set of keys and each corresponding value pair is deeply equal.
An additional constraint is that we must not use any external deep comparison function like lodash's _.isEqual()
.
Intuition
To determine if two given values are deeply equal, the solution takes a recursive approach:
- A straightforward comparison is made for primitive values and
null
. - If the values are arrays, it checks if they are the same length and then compares each corresponding pair of elements recursively.
- If the values are objects, it gets a list of keys for each, checks if those are the same length (to quickly rule out objects with different sets of keys), and then recursively checks deeply equality for each value associated with each key in the lists.
This approach works by systematically breaking down complex structures (arrays and objects) into simpler problems that are essentially similar to the base case: comparing primitives. Recursion is used to apply the same logic at every level of nested structures, allowing for a consistent method of comparison throughout the entire depth of the data.
Solution Approach
The solution provided above employs recursion to implement deep equality checking between two values, o1
and o2
. The recursion acts as a control flow mechanism that allows the function to traverse deeper into arrays and object properties if necessary. Here are the key parts of the algorithm with explanations:
-
Base Case for Primitives and
null
: Primitive values andnull
can be checked with a strict equality===
comparison. Ifo1
isnull
or not an object, the function returns the result ofo1 === o2
. -
Type Checking: The algorithm quickly rules out different types by checking whether
o1
ando2
are of the same type. This is done using thetypeof
operator. -
Array Check and Comparison: The function needs to identify if the values being compared are arrays, which is achieved using
Array.isArray(o1)
andArray.isArray(o2)
. If one is an array and the other is not, they can't be deeply equal, so the function returnsfalse
. -
Deep Array Equality: Once it's determined that both
o1
ando2
are arrays, the lengths are compared. If the lengths don't match, the arrays can't be deeply equal. If the lengths are the same, the function iteratively compares each element pair using a for-loop while recursively callingareDeeplyEqual
to check the deep equality of each pair. -
Object Property Equality: If
o1
ando2
are not arrays, the code treats them as objects. It retrieves the keys from both objects withObject.keys
and compares the lengths of these keys arrays. If they're unequal, it can immediately returnfalse
. Otherwise, it iterates over the keys of one object, callingareDeeplyEqual
recursively for each corresponding value pair. -
Recursive Descent: The recursion is essential to the algorithm, allowing the equality check to be valid no matter how deeply nested the arrays or objects are. Each call to
areDeeplyEqual
within the loops ensures arrays and objects nested within other arrays and objects are also subjected to this thorough checking.
Given the recursive nature of the function, it implicitly utilizes the call stack as its data structure, each frame holding context for a comparison at a particular level of depth. The patterns here are typical for recursive tree/graph traversal, though simplified because the inputs are limited to arrays and objects without cycles (thanks to the JSON restriction).
No additional data structures are utilized since the structure of the original inputs is directly traversed without the need for transformation or storage of intermediate results beyond what the recursive calls do inherently.
This implementation ensures a full deep equality check in line with the requirements outlined in the problem description.
Ready to land your dream job?
Unlock your dream job with a 2-minute evaluator for a personalized learning plan!
Start EvaluatorExample Walkthrough
Let's consider an example where we have two values that we want to compare using the solution approach described above. Our first value o1
is an object, and the second value o2
is another object. We want to determine if they are deeply equal.
1let o1 = { 2 a: 1, 3 b: [1, 2, {c: 3}], 4 d: {e: {f: 4}} 5}; 6 7let o2 = { 8 a: 1, 9 b: [1, 2, {c: 3}], 10 d: {e: {f: 4}} 11};
Both o1
and o2
look the same at first glance, but we need to check each element in detail.
-
Base Case for Primitives and
null
:- The algorithm starts by checking
o1.a
ando2.a
, which are both primitive values. - Using the
===
operator, we get that 1 === 1, which is true.
- The algorithm starts by checking
-
Type Checking:
- Next,
o1.b
ando2.b
are both arrays, so their types match (typeof o1.b
returns "object", as it does foro2.b
because arrays are objects in JavaScript).
- Next,
-
Array Check and Comparison:
- Since both are arrays, we use
Array.isArray(o1.b)
andArray.isArray(o2.b)
to confirm.
- Since both are arrays, we use
-
Deep Array Equality:
- We compare the lengths of
o1.b
ando2.b
which are both 3, so we move forward. - Then we check each element at the corresponding indexes:
- First pair:
o1.b[0] === o2.b[0]
(1 === 1 is true). - Second pair:
o1.b[1] === o2.b[1]
(2 === 2 is true). - Third pair: They are objects
{c: 3}
, so the algorithm checks them recursively. - During this recursive step, the third pair passes the base case for primitives (
o1.b[2].c === o2.b[2].c
which is true).
- First pair:
- We compare the lengths of
-
Object Property Equality:
- Lastly,
o1.d
ando2.d
are objects, not arrays. The keys ("e") and their lengths match, so we proceed to their values. o1.d.e
ando2.d.e
are also objects, triggering another recursion. They have the same set of keys "f", and their values are primitive, soo1.d.e.f === o2.d.e.f
(4 === 4 is true).
- Lastly,
-
Recursive Descent:
- Throughout the comparison, the
areDeeplyEqual
function calls itself for each pair of values that aren't strictly equal primitives, which includes objects and arrays regardless of their nesting level.
- Throughout the comparison, the
Since all checks passed without returning false
, o1
and o2
are determined to be deeply equal by the implementation.
Solution Implementation
1def are_deeply_equal(value1, value2):
2 """
3 Determines if two values are deeply equal.
4 Handles comparisons of primitives, arrays, and objects.
5 """
6 # Check for direct equality or if both are 'None'
7 if value1 is None or not isinstance(value1, (list, dict)):
8 return value1 == value2
9
10 # If types are different, they are not equal
11 if type(value1) != type(value2):
12 return False
13
14 # If values are lists (similar to JavaScript arrays), compare their elements
15 if isinstance(value1, list):
16 # Lists of different lengths are not equal
17 if len(value1) != len(value2):
18 return False
19
20 # Recursively compare each element of the list
21 for index in range(len(value1)):
22 if not are_deeply_equal(value1[index], value2[index]):
23 return False
24 return True
25
26 elif isinstance(value1, dict):
27 # Handle comparison of dictionary (object in JavaScript) items
28
29 # Objects with different numbers of keys are not equal
30 if len(value1) != len(value2):
31 return False
32
33 # Recursively compare each item of the dictionary
34 for key in value1:
35 if key not in value2 or not are_deeply_equal(value1[key], value2[key]):
36 return False
37
38 # All properties match, so the objects are deeply equal
39 return True
40
41 # For all other types that are not list or dict, check for direct equality
42 return value1 == value2
43
1import java.util.Arrays;
2import java.util.Objects;
3
4public class DeepEquality {
5
6 // Determines if two values are deeply equal.
7 // Handles comparisons of primitives, arrays, and objects.
8 public static boolean areDeeplyEqual(Object value1, Object value2) {
9 // Check for direct equality or if both values are null
10 if (value1 == value2) {
11 return true;
12 }
13
14 // If one is null and the other is not, they cannot be equal
15 if (value1 == null || value2 == null) {
16 return false;
17 }
18
19 // Check if types are different, then values are not equal
20 if (value1.getClass() != value2.getClass()) {
21 return false;
22 }
23
24 // If values are arrays, compare their elements
25 if (value1.getClass().isArray()) {
26 // Compare arrays deeply, considering multi-dimensional cases
27 return Arrays.deepEquals(new Object[]{value1}, new Object[]{value2});
28 } else if (value1 instanceof Iterable) {
29 // This if clause could handle the comparison of Iterable objects (like Lists)
30 // For simplicity, this feature is not implemented in this example
31 // Implement Iterable comparison logic here if required
32 return false; // Placeholder, replace with actual comparison code
33 } else if (value1 instanceof Objects) {
34 // Handle comparison of object properties
35 // For simplicity, this feature is not implemented in this example
36 // Implement custom object comparison logic here if required
37 return Objects.equals(value1, value2); // Placeholder, replace with deep comparison
38 } else {
39 // For non-array and non-Iterable objects, check equality
40 return value1.equals(value2);
41 }
42 }
43
44 public static void main(String[] args) {
45 // Example usage of areDeeplyEqual method
46 int[] array1 = {1, 2, 3};
47 int[] array2 = {1, 2, 3};
48 System.out.println("Are the arrays deeply equal? " + areDeeplyEqual(array1, array2));
49 }
50}
51
1#include <iostream>
2#include <vector>
3#include <unordered_map>
4
5// Helper function to compare two vectors deeply.
6// Returns true if given vectors are deeply equal; false otherwise.
7bool AreVectorsDeeplyEqual(const std::vector<any>& vector1, const std::vector<any>& vector2);
8
9// Helper function to compare two maps deeply.
10// Returns true if given maps are deeply equal; false otherwise.
11bool AreMapsDeeplyEqual(const std::unordered_map<std::string, any>& map1, const std::unordered_map<std::string, any>& map2);
12
13// Determines if two values are deeply equal.
14// Handles comparisons of primitives, vectors, and maps.
15bool AreDeeplyEqual(const any& value1, const any& value2) {
16 // Check for direct equality or if both are null
17 if (value1.IsNull() || !value1.IsObject()) {
18 return value1 == value2;
19 }
20
21 // If types are different, they are not equal
22 if (value1.Type() != value2.Type()) {
23 return false;
24 }
25
26 // Check if both values are vectors (and subsequently both are tuples)
27 if (value1.IsVector() != value2.IsVector()) {
28 return false;
29 }
30
31 // If values are vectors, compare their elements
32 if (value1.IsVector()) {
33 return AreVectorsDeeplyEqual(value1.AsVector(), value2.AsVector());
34 } else {
35 // Handle comparison of map properties
36 return AreMapsDeeplyEqual(value1.AsMap(), value2.AsMap());
37 }
38}
39
40bool AreVectorsDeeplyEqual(const std::vector<any>& vector1, const std::vector<any>& vector2) {
41 // Vectors of different lengths are not equal
42 if (vector1.size() != vector2.size()) {
43 return false;
44 }
45 // Recursively compare each element of the vector
46 for (size_t index = 0; index < vector1.size(); ++index) {
47 if (!AreDeeplyEqual(vector1[index], vector2[index])) {
48 return false;
49 }
50 }
51 return true;
52}
53
54bool AreMapsDeeplyEqual(const std::unordered_map<std::string, any>& map1, const std::unordered_map<std::string, any>& map2) {
55 // Maps with different numbers of keys are not equal
56 if (map1.size() != map2.size()) {
57 return false;
58 }
59
60 // Recursively compare each corresponding property of the map
61 for (const auto& pair : map1) {
62 const auto& key = pair.first;
63 if (map2.find(key) == map2.end() || !AreDeeplyEqual(map1.at(key), map2.at(key))) {
64 return false;
65 }
66 }
67
68 // All properties match, so the maps are deeply equal
69 return true;
70}
71
72// Note: 'any' is a placeholder here and would need to be defined or replaced with an appropriate type system
73// capable of handling different types (e.g., boost::any, std::variant from C++17, or a custom any class).
74
1// Determines if two values are deeply equal.
2// Handles comparisons of primitives, arrays, and objects.
3function areDeeplyEqual(value1: any, value2: any): boolean {
4 // Check for direct equality or if both are null
5 if (value1 === null || typeof value1 !== 'object') {
6 return value1 === value2;
7 }
8
9 // If types are different, they are not equal
10 if (typeof value1 !== typeof value2) {
11 return false;
12 }
13
14 // Check if both values are arrays (and subsequently both are tuples)
15 if (Array.isArray(value1) !== Array.isArray(value2)) {
16 return false;
17 }
18
19 // If values are arrays, compare their elements
20 if (Array.isArray(value1)) {
21 // Arrays of different lengths are not equal
22 if (value1.length !== value2.length) {
23 return false;
24 }
25 // Recursively compare each element of the array
26 for (let index = 0; index < value1.length; index++) {
27 if (!areDeeplyEqual(value1[index], value2[index])) {
28 return false;
29 }
30 }
31 return true;
32 } else {
33 // Handle comparison of object properties
34 const keys1 = Object.keys(value1);
35 const keys2 = Object.keys(value2);
36
37 // Objects with different numbers of keys are not equal
38 if (keys1.length !== keys2.length) {
39 return false;
40 }
41
42 // Recursively compare each corresponding property of the object
43 for (const key of keys1) {
44 if (!areDeeplyEqual(value1[key], value2[key])) {
45 return false;
46 }
47 }
48
49 // Ensure that value2 does not have extra properties
50 for (const key of keys2) {
51 if (!value1.hasOwnProperty(key)) {
52 return false;
53 }
54 }
55
56 // All properties match, so the objects are deeply equal
57 return true;
58 }
59}
60
Time and Space Complexity
Time Complexity
The time complexity of the areDeeplyEqual
function is O(N)
, where N
represents the total number of elements or properties within the objects (or arrays) being compared. This analysis is based on the fact that each element or property must be visited once to compare them. In the case of arrays, the comparison happens sequentially through iteration, and for objects, it happens by cycling through all the keys.
However, the worst-case time complexity can be more accurately described as O(N * K)
, where N
is the number of elements or properties in the objects and K
is the average depth of nested structures within those objects. Each recursive call to areDeeplyEqual
for nested objects or arrays adds an additional layer of complexity that is linear to the depth K
of the objects.
Space Complexity
The space complexity can also be seen as O(K)
, due to the recursion stack that grows in proportion to the depth of the nested structures (arrays or objects). Each recursive call takes up some space on the call stack, and in the case of deeply nested objects or arrays, the number of recursive calls corresponds to the depth of those structures.
Consider the classic dynamic programming of fibonacci numbers, what is the recurrence relation?
Recommended Readings
LeetCode Patterns Your Personal Dijkstra's Algorithm to Landing Your Dream Job The goal of AlgoMonster is to help you get a job in the shortest amount of time possible in a data driven way We compiled datasets of tech interview problems and broke them down by patterns This way we
Recursion Recursion is one of the most important concepts in computer science Simply speaking recursion is the process of a function calling itself Using a real life analogy imagine a scenario where you invite your friends to lunch https algomonster s3 us east 2 amazonaws com recursion jpg You first
Runtime Overview When learning about algorithms and data structures you'll frequently encounter the term time complexity This concept is fundamental in computer science and offers insights into how long an algorithm takes to complete given a certain input size What is Time Complexity Time complexity represents the amount of time