树——017树的子结构

1,问题描述

输入两棵二叉树A,B,判断B是不是A的子结构。(ps:我们约定空树不是任意一个树的子结构)

2,解题思路(转自:完美风暴4

(1)要查找树A中是否存在和树B结构一样的子树,我们可以分两步:第一步在树A中找到和B的根节点的值一样的结点R,第二步再判断树A中以R为根结点的子树是不是包含和树B一样的结构。

(2)以上面的两棵树为例来详细分析这个过程。首先我们试着在树A中找到值为8的结点。从树A的根节点开始遍历,我们发现它的根节点的值就是8。接着我们就去判断树A的根节点下面的子树是不是含有和树B一样的结构。在树A中,根节点的左子结点的值是8,而树B的根节点的左子结点是9,对应的两个结点不同。

(3)因此我们仍然要遍历A,接着查找值为8的结点。我们在树的第二层中找到一个值为8 的结点,然后进行第二步的判断,即判断这个结点下面的子树是否含有和树B一样结构的子树。于是我们遍历这个结点下面的子树,先后得到两个子节点9和2,这和树B的结构完全相同。此时我们在树A中找到了一个和树B的结构一样的子树,因此树B和树A的子结构。

在面试的时候,我们一定要检查边界条件,即检查空指针。当树A或树B为空的时候,定义相同的输出。如果没有检查并做响应的处理,程序非常容易崩溃,这是面试的时候非常忌讳的事情。

3,源码

/**
public class TreeNode {
    int val = 0;
    TreeNode left = null;
    TreeNode right = null;

    public TreeNode(int val) {
        this.val = val;

    }

}
*/
public class Solution {
    public boolean HasSubtree(TreeNode root1,TreeNode root2) {
        if(root1 == null || root2 == null) return false;
        boolean flag = false;
        if(root1.val == root2.val) {
            flag = equalTree(root1.left, root2.left) ? 
                equalTree(root1.right, root2.right) : false;
        }
        if(!flag) {
            flag = HasSubtree(root1.left, root2) ? true : HasSubtree(root1.right, root2);
        }
        return flag;

    }

    public boolean equalTree(TreeNode root1, TreeNode root2) {
        if(root1 == null && root2 != null) return false;
        if(root2 == null) return true;
        if(root1.val == root2.val) {
            return equalTree(root1.left, root2.left) && equalTree(root1.right, root2.right);
        }else {
            return false;
        }
    }
}

4,番外篇

另一种思想:先将两个二叉树进行中序遍历,再进行字符串匹配,以下是源码(源码有问题,待解决。。。。em。。。。觉得思路是没问题的,但是找不到源码中哪出了问题)

import java.util.ArrayList;	
//方法一:先中序遍历,再字符串匹配
public class Solution {
    public boolean HasSubtree(TreeNode root1,TreeNode root2) {
        if(root1 == null || root2 == null) return false;
        ArrayList<Integer> array1 = inorderTraversal(root1);
        ArrayList<Integer> array2 = inorderTraversal(root2);
        int n = array1.size();
        int m = array2.size();
        boolean flag = false;
        if(m<=n) {
        	flag = HasSubSequence(array2,m,array1,n);
        }else {
        	flag = HasSubSequence(array1,n,array2,m);
        }
        
        return flag;
    }
    
//	    //中序遍历方法一(需要返回遍历序列的情况下):空间复杂度太高
//	    public ArrayList<Integer> centerSearch(TreeNode root){
//	        ArrayList<Integer> array = new ArrayList<Integer>();
//	        if(root.left != null){
//	            array.addAll(centerSearch(root.left));
//	        }else{
//	            array.add(root.val);
//	            if(root.right != null) {
//	                array.addAll(centerSearch(root.right));
//	            }
//	        }
//	        return array;
//	    }
    
    //中序遍历方法二(需要返回遍历序列的情况下):推荐使用
    public ArrayList<Integer> inorderTraversal(TreeNode root) {
    	ArrayList<Integer> resultList = new ArrayList<Integer>();
    	if(root==null) return resultList;
    	creatSequence(resultList, root);
    	return resultList;
    };
    
    public void creatSequence(ArrayList<Integer> resultList, TreeNode root) {
    	if(root.left != null) creatSequence(resultList, root.left);
    	resultList.add(root.val);
    	if(root.right != null) creatSequence(resultList, root.right);
    	return;
    }
    
    public void creatSequence2(ArrayList<Integer> resultList, TreeNode root) {
    	if(root==null) return;
    	creatSequence2(resultList, root.left);
    	resultList.add(root.val);
    	creatSequence2(resultList, root.right);
    }
    
    
    
    public boolean HasSubSequence(ArrayList<Integer> array1, int m, 
    		ArrayList<Integer> array2, int n) {
    	boolean flag = false;
    	int count = 0;
    	int j = 0;
    	for(int i = 0; i<n; i++) {
    		if(array2.get(i) != array1.get(j)) {
    			j = 0;
    			if(count>0) {
    				i--;
    				count=0;
    			}
    		}else {
    			count++;
    			j++;
    			if(count == m) {
    				flag = true;
                    break;
    			}
    		}
    	}
    	
    	return flag;
    }
}

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值