二叉树:判断二叉树是否另一棵二叉树的子树

    给定两个二叉树s和t,判断t是否为s的子树。

    方法一:进行对比时,s不是从根节点开始的,而t是从根节点开始的,因此,可以先计算t的高度,然后在s中找出所有高度和t相同的子树,然后对比它们是否相同即可。

class Solution {
    int height = 0;
    LinkedList<TreeNode> queue = new LinkedList<TreeNode>();
    public boolean isSubtree(TreeNode s, TreeNode t) {
        height = height(t);
        height(s);
        while(!queue.isEmpty()){
        	TreeNode root = queue.poll();
        	if(isSame(root, t)) 
        		return true;
        }
        return false;
    }
    public int height(TreeNode root){
    	if(root == null)
    		return 0;
    	int h = Math.max(height(root.left), height(root.right)) + 1;
    	if(height == h)    //计算t的高度时,height=0,肯定不会相等。计算s的高度时,将高度等于height的所有节点装进队列。
    		queue.offer(root);
    	return h;
    }
    public boolean isSame(TreeNode s, TreeNode t){
    	if(s == null || t == null)
    		return s == t;
    	return isSame(s.left, t.left) && isSame(s.right, t.right) && s.val == t.val;
    }
}

    方法二:直接对s的左子树和右子树进行递归判断,强制循环所有节点。

    这种方法时间复杂度较高,176个测试案例,方法一的时间是16ms,方法二是26ms

public class Solution {
    public boolean isSubtree(TreeNode s, TreeNode t) {
        if (s == null) return false;
        if (isSame(s, t)) return true;
        return isSubtree(s.left, t) || isSubtree(s.right, t);
    }
    
    private boolean isSame(TreeNode s, TreeNode t) {
        if (s == null && t == null) return true;
        if (s == null || t == null) return false;
        
        if (s.val != t.val) return false;
        
        return isSame(s.left, t.left) && isSame(s.right, t.right);
    }
}

    方法三:把s和t转化成字符串,直接判断s是否包含t,虽然这种方法看起来是O(n),但实际上涉及到栈和字符串操作,测试案例的时间为39ms。

 public boolean isSubtree(TreeNode s, TreeNode t) {
        String spreorder = generatepreorderString(s); 
        String tpreorder = generatepreorderString(t);
        
        return spreorder.contains(tpreorder) ;
    }
    public String generatepreorderString(TreeNode s){
        StringBuilder sb = new StringBuilder();
        Stack<TreeNode> stacktree = new Stack();
        stacktree.push(s);
        while(!stacktree.isEmpty()){
           TreeNode popelem = stacktree.pop();
           if(popelem==null)
              sb.append(",#"); // Appending # inorder to handle same values but not subtree cases
           else      
              sb.append(","+popelem.val);
           if(popelem!=null){
                stacktree.push(popelem.right);    
                stacktree.push(popelem.left);  
           }
        }
        return sb.toString();
    }

判断一个二叉树是否为另一棵树的子树,可以采用递归的方法,先判断当前根节点是否相同,若相同则递归判断左右子树是否相同,若不同则递归判断二叉树的左右子树是否为目标树的子树。 具体实现步骤如下: 1. 判断当前节点是否为空,若为空则返回 false。 2. 判断当前节点是否与目标树的根节点相同,若相同则递归判断左右子树是否相同。 3. 若当前节点与目标树的根节点不同,则递归判断二叉树的左右子树是否为目标树的子树。 具体实现代码如下(假设 TreeNode 为二叉树节点的数据结构): ``` bool isSubtree(TreeNode* s, TreeNode* t) { if (!s) return false; // 若当前节点为空,返回 false if (isSameTree(s, t)) return true; // 若当前节点与目标树根节点相同,递归判断左右子树是否相同 return isSubtree(s->left, t) || isSubtree(s->right, t); // 若不同,则递归判断二叉树的左右子树是否为目标树的子树 } bool isSameTree(TreeNode* p, TreeNode* q) { if (!p && !q) return true; // 若两个节点均为空,返回 true if (!p || !q) return false; // 若其中一个节点为空,返回 false if (p->val != q->val) return false; // 若两个节点的值不同,返回 false return isSameTree(p->left, q->left) && isSameTree(p->right, q->right); // 递归判断左右子树是否相同 } ``` 其中 isSameTree 函数用于判断两个节点以及它们的子树是否相同。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值