333. Largest BST Subtree
nlgn就不说了。。说n的方法。
这个题做了好久。
一开始想到的是post-order traversal.
左右都是BST,然后自己也是BST,返还长度是左+右+自己(1)。
左右其中一个不是,或者自己不是的话,返还-1.
第一次尝试忽略了跳级问题,判断自己能不能和左右子树组成BST的情况是,自己必须比左子树的最大值大,比右子树的最小值小。。所以害的自己造个变量记录最大最小值。
提交的时候改了半天,主要问题在于更新最大最小但是会后,既要以子树更新,又要以自己的VAL更新,一开始忽略了其中一种。。
public class Solution
{
public class MaxMin
{
int max = Integer.MIN_VALUE;
int min = Integer.MAX_VALUE;
public MaxMin(int max, int min)
{
this.max = max;
this.min = min;
}
}
int a = 1;
public int largestBSTSubtree(TreeNode root)
{
if(root == null) return 0;
MaxMin r = new MaxMin(Integer.MIN_VALUE,Integer.MAX_VALUE);
int res = helper(root,r);
a = Math.max(res,a);
return a;
}
public int helper(TreeNode root, MaxMin m)
{
if(root == null) return 0;
if(root.left == null && root.right == null)
{
m.max = Math.max(m.max,root.val);
m.min = Math.min(m.min,root.val);
return 1;
}
MaxMin m1 = new MaxMin(m.max,m.min);
MaxMin m2 = new MaxMin(m.max,m.min);
int left = helper(root.left,m1);
int right = helper(root.right,m2);
int res = 1;
m.max = Math.max(m.max,root.val);
m.min = Math.min(m.min,root.val);
m.max = Math.max(Math.max(m.max,m1.max),m2.max);
m.min = Math.min(Math.min(m.min,m1.min),m2.min);
if(left != -1 && right != -1)
{
if(root.val < m2.min && root.val > m1.max)
{
res += left + right;
a = Math.max(res,a);
return res;
}
else return -1;
}
else return -1;
}
}
另一种N的尝试是,做一个in-order traversal,找其中的递增组合。不过仔细想想犹豫结构不确定,不一定是completed tree所以无法判定。。
二刷。
这个题做了他妈好久。
回头看一刷觉得太繁琐了,肯定有问题,但是又不想在基础上改。
像题目说的,直白的做法是每个NODE都call一次valid BST,这样是O(nlgn).
现在要求O(n),只能遍历一次。那肯定是bottom-up,用post-order traversal.
有一点要注意的是,subtree得到底才行,必须包含最下的leaves.
我们需要传到上面的信息有这么几个:
我是不是BST,以便于上面判断他是不是BST.
我作为BST的最大Size,以便于上面进行左右取舍,或相加(如果左右+自己都是BST)
我的取值区间,最大值和最小值。对于上面的parent node来说,他得比左支最大值大,比右支最小值小。
需要传递的信息有3个,自己建个新的class比较方便。
学到的一个tricky是用size正负来表示1),是否是BST。负数就说明我不是BST。
Time: O(n) DFS post-order
Space: O(n) for Stack in memory
public class Solution {
public class Node {
int max;
int min;
int num;
public Node(int max, int min, int num) {
this.max = max;
this.min = min;
this.num = num;
}
}
public int largestBSTSubtree(TreeNode root) {
return Math.abs(dfs(root).num);
}
public Node dfs(TreeNode root) {
if (root == null) return new Node(Integer.MIN_VALUE, Integer.MAX_VALUE, 0);
Node leftRes = dfs(root.left);
Node rightRes = dfs(root.right);
int curMin = Math.min(root.val, leftRes.min);
int curMax = Math.max(root.val, rightRes.max);
if (root.val <= leftRes.max || root.val >= rightRes.min || leftRes.num < 0 || rightRes.num < 0) {
//System.out.println(leftRes.num + " " + rightRes.num);
return new Node(curMax, curMin, Math.max(Math.abs(leftRes.num), Math.abs(rightRes.num)) * -1);
} else {
return new Node(curMax, curMin, leftRes.num + rightRes.num + 1);
}
}
}
2个判断不太好理解,着重说一下。
if (root.val <= leftRes.max || root.val >= rightRes.min || leftRes.num < 0 || rightRes.num < 0) {
return new Node(curMax, curMin, Math.max(Math.abs(leftRes.num), Math.abs(rightRes.num)) * -1);
}
这说明不符合BST规定。
root.val <= 左支最大值. root.val >= 右支最小值.
左支BST的SIZE < 0 或者 右支BST的Size < 0
这个时候返还的Node要注意。
curMax和curMin说实话其实无所谓,返还什么都行,因为第三个值要返还负数。
第三个值首先要确定返还负数,最后乘以-1。
判断前取绝对值Math.abs(leftRes.num)是代表:
子树最大subTree的size,不一定就是root.left的值,可能还要往下。
正负号代表当前子树(root.left)是否是BST.
else是判断成功,左右都是,自己也是,那么显然左+右+1就行了。
333. Largest BST Subtree的更多相关文章
- [LeetCode] 333. Largest BST Subtree 最大的二分搜索子树
Given a binary tree, find the largest subtree which is a Binary Search Tree (BST), where largest mea ...
- 333. Largest BST Subtree节点数最多的bst子树
[抄题]: Given a binary tree, find the largest subtree which is a Binary Search Tree (BST), where large ...
- [leetcode]333. Largest BST Subtree最大二叉搜索树子树
Given a binary tree, find the largest subtree which is a Binary Search Tree (BST), where largest mea ...
- LeetCode 333. Largest BST Subtree
原题链接在这里:https://leetcode.com/problems/largest-bst-subtree/ 题目: Given a binary tree, find the largest ...
- 【LeetCode】333. Largest BST Subtree 解题报告(C++)
作者: 负雪明烛 id: fuxuemingzhu 个人博客:http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS 日期 题目地址:https://leetcod ...
- [LeetCode] Largest BST Subtree 最大的二分搜索子树
Given a binary tree, find the largest subtree which is a Binary Search Tree (BST), where largest mea ...
- Largest BST Subtree
Given a binary tree, find the largest subtree which is a Binary Search Tree (BST), where largest mea ...
- Leetcode: Largest BST Subtree
Given a binary tree, find the largest subtree which is a Binary Search Tree (BST), where largest mea ...
- [Locked] Largest BST Subtree
Largest BST Subtree Given a binary tree, find the largest subtree which is a Binary Search Tree (BST ...
随机推荐
- centos 7 samba相关命令
1.安装相关包 yum install samba samba-client samba-common 2.启动smb的命令 systemctl enable smb.service systemct ...
- AppiumDriver 运行app启动基本参数
记录一下 DesiredCapabilities capabilities = new DesiredCapabilities(); capabilities.setCapability(Mobile ...
- sencha touch中用来格式化日期的字符串参数
- HTML5的Server-Sent Events (SSE)
HTML5有一个Server-Sent Events(SSE)功能,允许服务端推送数据到客户端.(通常叫数据推送).我们来看下,传统的WEB应用程序通信时的简单时序图: 现在Web App中,大都有A ...
- Linux系统上使用php获取apk信息
最近在做一个apk商城,需要在用户上传了apk之后系统自动读取apk信息(包名,版本号等),后台语言使用的是php,需要php去调用系统的aapt命令去读取apk信息,在Linux系统上安装aapt的 ...
- 理解Python的*args, **kwargs
1 # coding=utf-8 2 def speak(*args, **kwargs): 3 print args, kwargs 4 5 6 a = 1 7 b = 2 8 c = 3 9 sp ...
- Java高精度学习第三弹——ACM中使用JAVA的详细介绍
Chapter I. Java的优缺点各种书上都有,这里只说说用Java做ACM-ICPC的特点: (1) 最明显的好处是,学会Java,可以参加Java Challenge . (2) 对于熟悉C/ ...
- iOS 检测版本更新
如果我们要检测app版本的更新,那么我们必须获取当前运行app版本的版本信息和appstore 上发布的最新版本的信息. 当前运行版本信息可以通过info.plist文件中的bundle versio ...
- 单例-b
这个比较老了,是mrc 里面的 此例以模仿Apple官方文档的单例写出来的.但是一直有一个非常不明白的地方,就是alloc与allocWithZone:的重载中,为什么要return [[self c ...
- 转:推荐!国外程序员整理的 C++ 资源大全
原文来自于:http://blog.jobbole.com/78901/ 关于 C++ 框架.库和资源的一些汇总列表,由 fffaraz 发起和维护. 内容包括:标准库.Web应用框架.人工智能.数据 ...