点分治的时间复杂度为O(NlogN)。
由于每次都是找重心,所以处理完一个大小为N的树后,每个子树的大小最大都为N/2,所以最多分治NlogN层,每层都是N
所以是O(NlogN)。
【具体流程】
1,选取一个点,将无根树变成有根树为了使每次的处理最优,我们通常要选取树的重心。
何为“重心”,就是要保证与此点连接的子树的节点数最大值最小,可以防止被卡。
重心求法:
1。dfs一次,算出以每个点为根的子树大小。
2。记录以每个节点为根的最大子树大小
3。判断:如果以当前节点为根更优,就更新当前根。
void getroot(int v,int fa)
{
son[v] = 1; f[v] = 0;//f记录以v为根的最大子树的大小
for(int i = head[v];i;i=e[i].next)
if(e[i].to != fa && !vis[e[i].to]) {
getroot(e[i].to,v);//递归更新
son[v] += son[e[i].to];
f[v] = max(f[v],son[e[i].to]);//比较每个子树
}
f[v] = max(f[v],sum-son[v]);//别忘了以v父节点为根的子树
if(f[v] < f[root]) root = v;//更新当前根
}
2、处理连通块中通过根节点的路径。 (注意,是通过根节点的路径,所以后面要去掉同一子树内部的路径,即去重。)
3、标记根节点(相当于处理后,将根节点从子树中删除)。4、递归处理当前点为根的每棵子树。
int solve(int v)
{
vis[v] = 1;//标记
for(int i = head[v];i;i=e[i].next)
if(!vis[e[i].to]) {
root = 0;
sum = son[e[i].to];
getroot(e[i].to,v);
solve(root);//递归处理下一个连通块
}
}
int main()
{
sum = f[0] = n;//初始化
root = 0;
getroot(1,0);//找重心
solve(root);//点分治
}