求LCA,N=1e6,原空間限制8MBnode
求LCA須要深度,須要跳躍必定距離的祖先,須要父節點ios
把一個整數壓成3個char,f[]存父節點ui
g[],深度爲奇數的點存往上跳576步能到的點,深度爲偶數的點存深度spa
若是深度爲奇數的點要求它的深度,求他父節點的深度+13d
若是深度爲偶數的點要求它往上跳576步的祖先,先往上跳一步,變成奇數深度code
如何求往上跳576步的祖先?blog
一、往上跳6次1步,g就存的往上跳6步的祖先get
二、往上跳6次6步,g就存的往上跳36步的祖先io
三、往上跳4次36步,g就存的往上跳144步的祖先class
四、往上跳4次144步,g就存的往上跳576步的祖先
還有,不能開iostream庫,否則爆空間。。。
#include<cstdio> using namespace std; #define N 1000001 const int D=576; struct node { char a,b,c; }f[N],g[N]; inline int F(int i) { return ((f[i].a<<7)+f[i].b<<7)+f[i].c; } inline int G(int i) { return ((g[i].a<<7)+g[i].b<<7)+g[i].c; } inline int d(int i) { return G(i)<N ? G(i):G(F(i))+1; } void read(int &x) { x=0; char c=getchar(); while(c<'0'||c>'9') c=getchar(); while(c>='0'&&c<='9') { x=x*10+c-'0'; c=getchar(); } } node turn(int x) { node y; y.a=x>>14; y.b=(x>>7)&127; y.c=x&127; return y; } int lca(int x,int y) { int u; if(d(x)<d(y)) u=x,x=y,y=u; int dis=d(x)-d(y); while(dis>=D) { if(G(x)<N) x=F(x),dis--; else x=G(x)-N,dis-=D; } while(dis) x=F(x),dis--; while(x!=y) { if(G(x)<N||G(x)==G(y)) x=F(x),y=F(y); else x=G(x)-N,y=G(y)-N; } return x; } int main() { freopen("squirrel.in","r",stdin); freopen("squirrel.out","w",stdout); int n,m; read(n); read(m); g[0]=turn(N); int x; for(int i=2;i<=n;++i) { read(x); f[i]=turn(x),g[i]=turn(G(F(i))+1); } for(int i=n;i;i--) if(G(i)&1) { x=i; for(int j=1;j<=6;++j) x=F(x); g[i]=turn(x+N); } for(int i=n;i;i--) if(G(i)>N) { x=i; for(int j=1;j<=6;++j) x=G(x)-N; g[i]=turn(x+N); } for(int i=n;i;i--) if(G(i)>N) { x=i; for(int j=1;j<=4;++j) x=G(x)-N; g[i]=turn(x+N); } for(int i=n;i;i--) if(G(i)>N) { x=i; for(int j=1;j<=4;++j) x=G(x)-N; g[i]=turn(x+N); } int y; while(m--) { read(x); read(y); printf("%d\n",lca(x,y)); } }