首先考虑最暴力的暴力,那就是对于每颗子树都检验一次,然后求一个最大值,那么这个时间复杂度大约是 的,所以考虑优化检验方式
那么主要就是看 函数的实现了
bool Check(int L,int R) { if (L == -1 && R == -1)return 1; return (L != -1 && R != -1 && V[L] == V[R] && Check(Son[L][1],Son[R][2]) && Check(Son[L][2],Son[R][1])); }
那么对于完全二叉树,显然这个函数每次最多会向下跳 次,
如果不是完全二叉树,显然它能跳的次数就更少了,因为又特别强有力的权值剪枝和子树大小剪枝
最后,如果返回的是 true
那么就可以用这个子树的 size
去更新一下答案好像就可以了
所以总时间复杂度为
#include <cstdio> #define max(a, b) (a > b ? a : b) using namespace std; const int Maxn = 1e6+10; int N, V[Maxn], Son[Maxn][3]; int Size[Maxn], Ans(1); inline int Read() { int X(0), T(1); char O = getchar(); while (O < '0' || O >'9') { if (O == '-')T = -1; O = getchar(); } while (O >= '0' && O <= '9') { X = (X << 1) + (X << 3) + (O ^ 48); O = getchar(); } return X * T; } bool Check(int L, int R) { if (L == -1 && R == -1)return 1; return (L != -1 && R != -1 && V[L] == V[R] && Check(Son[L][1],Son[R][2]) && Check(Son[L][2],Son[R][1])); } int DFS(int X) { Size[X] = 1; if (Son[X][2] == -1 && Son[X][1] == -1)return Size[X]; if (Son[X][1] > 0)Size[X] += DFS(Son[X][1]); if (Son[X][2] > 0)Size[X] += DFS(Son[X][2]); if (Size[Son[X][1]] != Size[Son[X][2]] || V[Son[X][1]] != V[Son[X][2]])return Size[X]; if(Check(Son[X][1], Son[X][2]))Ans = max(Ans, Size[X]); return Size[X]; } int main() { N = Read(); for (int i = 1; i <= N; ++i)V[i] = Read(); for (int i = 1; i <= N; ++i)Son[i][1] = Read(), Son[i][2] = Read(); DFS (1); printf ("%d\n",Ans); return 0; }