Problem
Given the roots of two binary trees root
and subRoot
, return true
if there is a subtree of root
with the same structure and node values of subRoot
and false
otherwise.
A subtree of a binary tree tree
is a tree that consists of a node in tree
and all of this node’s descendants. The tree tree
could also be considered as a subtree of itself.
Examples
Example 1:
Input:
root = [3,4,5,1,2], subRoot = [4,1,2]
Output:
true
Example 2:
Input:
root = [3,4,5,1,2,null,null,null,null,0], subRoot = [4,1,2]
Output:
false
Solution
Lets denote tree with root
as p
and tree with subRoot
as q
.
Method 1 - Using Inorder and Postorder Traversal
Algorithm
- Perform inorder traversal of tree
p
and store it as stringinorderP
. - Perform inorder traversal of tree
q
and store it as stringinorderQ
. - Perform postorder traversal of tree
p
and store it as stringpostorderP
. - Perform postorder traversal of tree
q
and store it as stringpostorderQ
. - Tree q is a subtree of tree p if:
inorderP
containsinorderQ
(matches the inorder sequence)postorderP
containspostorderQ
(matches the postorder structure within the identified sequence)
Example
Lets look at example 1:
Lets look at inorder
:
inorderP
=1 4 2 3 5
inorderQ
=1 4 2
inorderP.contains(inorderQ)
istrue
Now, lets look at postorder
:
postorderP
=1 2 4 5 3
postorderQ
=1 2 4
postorderP.contains(postorderQ)
istrue
Code
Java
public boolean checkSubtree(Node rootA, Node rootB) {
String inOrderA = inOrder(rootA, "");
String inOrderB = inOrder(rootB, "");
String postOrderA = postOrder(rootA).toString();
String postOrderB = postOrder(rootB).toString();
return (inOrderA.toLowerCase().contains(inOrderB.toLowerCase()) && postOrderA.toLowerCase().contains(postOrderB.toLowerCase()));
}
Complexity
- ⏰ Time complexity:
O(|p|+|q|)
. Generating inorder and postorder traversal takeO(p)
andO(q)
, butcontains
may takeO(p)
assuming KMP algorithm. (#TODO link KMP algorithm). If we use brute forcecontains
, then, it will takeO(pq)
. - 🧺 Space complexity:
O(|p|+|q|)
Method 2 - Using Same Tree
We have already seen - Same Tree. Even though method 1 is faster asymptotically (if we use KMP), still that is not that clean.
See the example 2, leaf node in subRoot are not same as root, even though tree matches. So, subtrees have to be same after a somepoint.
Code:
// is q subtree of p
public boolean isSubtree(TreeNode p, TreeNode q) {
// null tree is always subtree
if (q == null) {
return true;
}
// we have already checked q, order of if's matter here
if(p == null) {
return false;
}
if (isSameTree(p, q)) {
return true;
}
return isSubtree(p.left, q) || isSubtree(p.right, q);
}
public boolean isSameTree(TreeNode p, TreeNode q) {
if (p == null && q == null) {
return true;
}
if (p == null || q == null) {
return false;
}
return p.val == q.val &&
isSameTree(p.left, q.left) &&
isSameTree(p.right, q.right);
}
Complexity
- ⏰ Time complexity:
O(p + q)
- 🧺 Space complexity:
O(NNNXXX)