题目描述
一棵有点权的有根树如果满足以下条件,则被轩轩称为对称二叉树:
1. 二叉树;
2. 将这棵树所有节点的左右子树交换,新树和原树对应位置的结构相同且点权相等。
下图中节点内的数字为权值,节点外的 𝑖𝑑 表示节点编号。
QQ截图20190312054152.png
现在给出一棵二叉树,希望你找出它的一棵子树,该子树为对称二叉树,且节点数最多,请输出这棵子树的节点数。
注意:只有树根的树也是对称二叉树。本题中约定,以节点 𝑇 为子树根的一棵“子树”指的是:节点 𝑇 和它的全部后代节点构成的二叉树。
输入格式
第一行一个正整数 n,表示给定的树的节点的数目,规定节点编号 1~n,其中节点 1 是树根。
第二行 n 个正整数,用一个空格分隔,第 i 个正整数 vi 代表节点 i 的权值。
接下来 n 行,每行两个正整数 li,ri ,分别表示节点 i 的左右孩子的编号。
如果不存在左 / 右孩子,则以 −1 表示。两个数之间用一个空格隔开。
输出格式
输出共一行,包含一个整数,表示给定的树的最大对称二叉子树的节点数。
数据范围
vi≤1000,n≤106
输入样例:
10
2 2 5 5 5 5 4 4 2 3
9 10
-1 -1
-1 -1
-1 -1
-1 -1
-1 2
3 4
5 6
-1 -1
7 8
输出样例:
3
分析
此题需用long long来存储,时间复杂度为O(n^2),所以我们得用inline 来加快函数的调取
C++代码
#include<bits/stdc++.h>
using namespace std;
const int N=1000010;
long long a[N],l[N],r[N],n,ans=0;
inline bool dfs(long long l2,long long r2){
if(l2==-1&&r2==-1)
return true;
if(l2==-1||r2==-1||a[l2]!=a[r2])
return false;
return dfs(l[l2],r[r2])&&dfs(l[r2],r[l2]);
}
inline long long get(long long x){
if(x==-1)
return 0;
else
return get(l[x])+get(r[x])+1;
}
int main(){
cin>>n;
for(long long i=1;i<=n;i++)
cin>>a[i];
for(long long i=1;i<=n;i++)
cin>>l[i]>>r[i];
for(long long i=1;i<=n;i++)
if(dfs(i,i))
ans=max(ans,get(i));
cout<<ans;
return 0;
}
这个做法虽然是暴力,但是复杂度好像$O(nlog_2n)$
唔,dfs是logn的时间复杂度,get也一样。并列取logn,大的循环一个n。