题目描述

给定两个非空二叉树 s 和 t,检验 s 中是否包含和 t 具有相同结构和节点值的子树。s 的一个子树包括 s 的一个节点和这个节点的所有子孙。s 也可以看做它自身的一棵子树。

暴力解法

从s的根节点开始遍历,查看该节点下的子树是否与t相同。方法是同步对s和t进行遍历,一旦出现s和t有不同(包括只有其中一个为NULL,或都不为NULL时value不同),就返回为false。如果最终返回给调用比较函数的地方是false,那么就继续为s的下一个节点重新遍历。

class Solution {
public:
bool isSameTree(TreeNode * sNow, TreeNode * tNow)
{
if(!sNow && !tNow) return true;
if((!sNow && tNow) || (sNow && !tNow) || (sNow->val != tNow->val))
return false;
//这里直接返回的是:当左子树和右子树全部相等时,就是匹配的。
//为什么没有比较本节点?因为假如本节点value不一样,就已经在上面一个if被返回false了,不会执行到最后的return
return isSameTree(sNow->left, tNow->left) && isSameTree(sNow->right, tNow->right);
} bool dfs(TreeNode * sNow, TreeNode * tNow)
{
if(!sNow) return false;
//由于||和&&的短路特性,||后面的表达式只有前面为false才会被运行
return isSameTree(sNow, tNow) || dfs(sNow->left, tNow) || dfs(sNow->right, tNow);
} bool isSubtree(TreeNode* s, TreeNode* t) {
bool isSub = dfs(s, t);
return isSub;
}
};

显然暴力解法的复杂度太高了。其时间复杂度为O(|s|*|t|),其中|s|和|t|是s树和t树的节点数量。空间复杂度方面,递归栈最大为O(max{ds, dt}),其中ds, dt分别是s和t的最大深度。

遍历后匹配字符串

为了解决暴力解法复杂度高的问题,一个容易想到的思路是首先对s和t前序遍历之后形成数组(字符串),然后再来进行字符串的匹配。

这样会出现一个显而易见的问题,当t是s的“一部分”而并非子树时,也就是t的“底端”比s多一些节点时,会出现匹配失误的情况。为了解决该问题,可以将叶节点左、右节点的NULL也插入到字符串中,这样可以保证匹配唯一。

遍历过后,就成为了一个字符串匹配问题。我们把较长的需要寻找子串的一个叫做文本串S(此处是s树得来的),匹配的目标叫做模式串P(此处是t树得来的)。

直接匹配

进行字符串匹配时,首先可以从暴力匹配想起。其思路是:对于S[i], P[j],如果匹配,则继续往后一位匹配。如果失配,则S退回到i = i-j+1,也就是与j开始匹配的后一位,P退回到j = 0,重新开始匹配过程。

此算法的浪费之处在于,假如失配时S[i-j+1]与S[i-j]并不相同,那么P重新从S[i-j+1]开始匹配时必然是失配的,在找到下一个匹配开始点前的操作都是重复的。

KMP算法匹配

KMP算法原理

KMP算法的核心在于利用了已经匹配过的信息。其关键在于加入了一个next数组。

next数组的意义是:next[k]表示从模式串的子串P[0]到P[k]中,相同前缀后缀的最大长度。例如:P = ABCDFABGF,那么next[6] = 2,即子串ABCDFAB的相同前缀后缀最长是AB,长度为2。

有了next数组,如果首位失配则S[i]后移(i++),如果非首位失配,则下次迭代只要从S[i]和P[next[j]]开始即可。因为在失配处的前面子串里,长度为next[j]的后缀是跟前缀一样的,这next[j]位必定是匹配的,无需再次迭代。

next数组求解

那么还剩下一个问题,next数组如何求解?

首先直观的方式是直接索引长度为1、2、3……的开头和结尾串,直到长度j-1的子串。其计算的重复性也是显而易见的。

较好的方法是使用动态规划。思考一下,当已知next[0...j]时,如何求出next[j+1]的值?

令k = next[j],即从开头算起长度为j的子串,最长的相同前缀后缀长度为k。此时有两种情况:

  1. p[k] = p[j]:即之前的前后缀再往后看一格,也是相同的。所以此时这个长度就增加了1,即next[j+1] = k + 1。
  2. p[k] ≠ p[j]:则令k = next[k],即在原本的最长相同前缀子串里再寻找它的最长相同前缀,重复第一步。意思是说,对于p[j+1],如果之前的相同前后缀再加一位是不相同的,那么再到这个前缀里去找,能不能找到?这样循环往复,直到最开始为止。
class Solution {
private:
int maxElement,lNULL, rNULL;
vector<int> sValues, tValues;
public:
void getMaxElement(TreeNode *p)
{
if(!p) return;
maxElement = max(maxElement, p->val);
getMaxElement(p->left);
getMaxElement(p->right);
} void traverse(TreeNode * p, vector<int> & stack)
{
if(!p) return;
stack.push_back(p->val); if(p->left == NULL)
stack.push_back(lNULL);
else
traverse(p->left, stack); if(p->right == NULL)
stack.push_back(rNULL);
else
traverse(p->right, stack);
} bool kmp()
{
int i = 0, j = 0;
int sLen = sValues.size(), tLen = tValues.size();
vector<int> next(tLen, 0); //首先计算next数组
for(int k = 1; k < tLen; k++)
{
int index = k;
while(index > 0)
{
if(tValues[k] == tValues[next[index - 1]])
{
next[k] = next[index - 1] + 1;
break;
}
else
{
index = next[index - 1];
}
}
if(index <= 0) next[k] = 0;
} //从头开始匹配字符串,失配后使用next数组重新尝试匹配
while(i < sLen && j < tLen)
{
if(j == 0)
{
while(sValues[i] != tValues[j] && i < sLen)
i++;
} if(sValues[i] == tValues[j])
{
i++;
j++;
}
else
{
if(j > 0)
j = next[j - 1];
else
j = 0;
}
}
if(j == tLen)
return true;
return false;
} bool isSubtree(TreeNode* s, TreeNode* t) {
maxElement = INT_MIN;
getMaxElement(s); //找出s和t中的最大值,分别+1 +2作为左右NULL值
getMaxElement(t);
lNULL = maxElement + 1;
rNULL = maxElement + 2; traverse(s, sValues); //对s,t进行前序遍历
traverse(t, tValues); return kmp();
}
};

看起来简单实际上却很牛的KMP算法:LeetCode572-另一棵树的子树的更多相关文章

  1. 听着好像很牛的特效——幽灵按钮DOM

    给大家分享一个听着好像很牛的东西——幽灵按钮,这个玩意对于艺术设计细胞在高中决定不在考试试卷上画画的我来说,实在不感冒.但是这个按钮的设计元素很流行,一个网页东西不做几个,光放上几个按钮就会显得很高端 ...

  2. iPhone不为人知的功能常用技巧,看完后才发现很多用iPhone的人实在是愧对乔布斯! - imsoft.cnblogs

    很多人花了四五千买部苹果,结果只用到四五百块钱的普通手机功能. iPhone不为人知的功能,常用技巧: 网上搜集整理的iPhone快捷键操作,虽然表面上iPhone按键只有一个HOME键,大部分操作都 ...

  3. 你很牛B,面试却没过,为什么?

    点击标题下「飞测」可快速关注 坚持的是分享,搬运的是知识,图的是大家的进步,没有收费的培训,没有虚度的吹水,喜欢就关注.转发(免费帮助更多伙伴)等来交流,想了解的知识请留言,给你带来更多价值,是我们期 ...

  4. 一个很牛的计算pi的c程序!

    C语言是面向过程的一种高级程序设计语言,它在世界范围内使用很广泛,而且很流行.很多大型的应用软件,基本上是用C语言所编写的.在对操作系统以及系统使用程序.需要对硬件进行操作的场合,C语言较其他的高级语 ...

  5. Javascript中几个看起来简单,却不一定会做的题

    Javascript作为前端开发必须掌握的一门语言,因为语言的灵活性,有些知识点看起来简单,在真正遇到的时候,却不一定会直接做出来,今天我们就一起来看看几道题目吧 题目1 var val = 'smt ...

  6. 零元学Expression Blend 4 - Chapter 31 看如何简单的把SampleData 绑进ListBox里

    原文:零元学Expression Blend 4 - Chapter 31 看如何简单的把SampleData 绑进ListBox里 前面几章连续讲到ListBox的运用,本章要讲得是如何简单的把Sa ...

  7. 不推荐别的了,IDEA 自带的数据库工具就很牛逼!

    MySQL 等数据库客户端软件市面上非常多了,别的栈长就不介绍了, 其实 IntelliJ IDEA 自带的数据库工具就很牛逼,不信你继续往下看. 本文以 IntelliJ IDEA/ Mac 版本作 ...

  8. KMP算法再解 (看毛片算法真是人如其名,哦不,法如其名。)

    KMP算法主要解决字符串匹配问题,其中失配数组next很关键: 看毛片算法真是人如其名,哦不,法如其名. 看了这篇博客,转载过来看一波: 原博客地址:https://blog.csdn.net/sta ...

  9. 很详尽KMP算法(厉害)

    作者:July时间:最初写于2011年12月,2014年7月21日晚10点 全部删除重写成此文,随后的半个多月不断反复改进.后收录于新书<编程之法:面试和算法心得>第4.4节中. 1. 引 ...

  10. 2020牛客寒假算法基础集训营2 J题可以回顾回顾

    2020牛客寒假算法基础集训营2 A.做游戏 这是个签到题. #include <cstdio> #include <cstdlib> #include <cstring ...

随机推荐

  1. Typora如何配置gitee图床

    转载自:https://mp.weixin.qq.com/s/5dPLbr2vFgL18XKL1Y05Og 要求: 1.Typora需要升级到最新版 2.需要安装nodejs PicGo软件下载地址: ...

  2. SC命令---安装、开启、配置、关闭windows服务 bat批处理

    一.直接使用cmd来进行服务的一些操作 1.安装服务 sc create test3 binPath= "C:\Users\Administrator\Desktop\win32srvDem ...

  3. redis cluster 6.2集群

    redis最新版本:redis-6.2.1.tar.gz 安装的版本是redis-6.0.3 采用的主机: djz-server-001 192.168.2.163 7001,7002,Admin@1 ...

  4. linux主机时间同步

    yum -y install ntpdate ntp && ntpdate cn.pool.ntp.org systemctl start ntpd.service && ...

  5. PAT (Basic Level) Practice 1029 旧键盘 分数 20

    旧键盘上坏了几个键,于是在敲一段文字的时候,对应的字符就不会出现.现在给出应该输入的一段文字.以及实际被输入的文字,请你列出肯定坏掉的那些键. 输入格式: 输入在 2 行中分别给出应该输入的文字.以及 ...

  6. 二叉树及其三种遍历方式的实现(基于Java)

    二叉树概念: 二叉树是每个节点的度均不超过2的有序树,因此二叉树中每个节点的孩子只能是0,1或者2个,并且每个孩子都有左右之分. 位于左边的孩子称为左孩子,位于右边的孩子成为右孩子:以左孩子为根节点的 ...

  7. Linux基础_1_简介

    Linux是什么 一款优秀的操作系统软件,特性是一切皆文件:一切设备皆文件!一切设备的设置皆修改配置文件!一切服务的搭建皆修改配置文件!(庞大的树形结构文件系统) 根据FHS标准,Linux目录有以下 ...

  8. 【Chrome浏览器】关闭触摸板双指滑动进行前进后退的功能

    痛点 Chrome浏览器使用过程中,当前页面经常会莫名其妙地退回到上一个浏览的页面. 当时真是一脸懵B(心里一万头草泥马呼啸而过~)!以为活见鬼了! 后来才发现浏览器左边,有一个幽灵般的淡蓝色箭头的出 ...

  9. Hadoop集群简单入门

    Hadoop集群搭建 自己配置Hadoop的话太过复杂了,因为自己着急学习,就使用了黑马的快照.如果小伙伴们也想的话可以直接看黑马的课程,快照的话关注黑马程序员公众号,输入Hadoop就能获取资料,到 ...

  10. HPL Study 2

    1.并行编程 (1)并行程序的逻辑: 1)将当前问题划分为多个子任务 2)考虑任务间所需要的通信通道 3)将任务聚合成复合任务 4)将复合任务分配到核上 (2)共享内存编程: 路障 ----> ...