Solution

这题的解法很妙啊... 考虑这三个点可能的形态: 令它们的重心为距离到这三个点都相同的节点, 则其中两个点分别在重心的两棵子树中, 且到重心的距离相等; 第三个点可能在重心的一棵不同于前两个点子树上, 也有可能在重心往上走可以到达的位置上.

定义数组\(f[i][j]\)表示在以\(i\)为根的子树下与\(i\)的距离为\(j\)的节点个数; \(g[i][j]\)表示在以\(i\)为根的子树下, 有多少个点对满足如下条件: 这个点对到它们LCA的距离相同, 我们假设其为\(d\), 则\(i\)到它们的LCA的距离为\(d - i\), 也就是说, 假如这两个点要找到一个在\(i\)上方的第三个点组成一组答案, 则第三个点到\(i\)的距离为\(j\).

考虑枚举每个点作为重心的情况. 进行一次DFS, 令\(u\)为当前点, \(v\)为\(u\)的一个子节点, 则有:

\[ans += \sum_i g[u][i] \times f[v][i - 1] + g[v][i] \times f[u][i - 1] \\
g[u][i] += g[v][i + 1] + f[u][i] \times f[v][i - 1] \\
f[u][i] += f[v][i - 1] \\
\]

然后我们发现这种方法的转移是\(O(n^2)\)的... 考虑如何优化: 我们注意到, 当一个点\(u\)计算其第一个子节点时, 可以直接将\(f[u][i]\)赋值为\(f[v][i - 1]\), \(g[u][i]\)赋值为\(g[v][i + 1]\), 因此在计算完这个子节点后, 直接对返回的数组指针进行位移就可以得到当前点的\(f\)和\(g\). 因此考虑采用按深度树链剖分的方法, 从重儿子处继承\(f\)和\(g\)数组.

时间复杂度: \(O(n)\). 为什么? 不会证. 以后学了长链剖分再填坑吧.

由于数组是动态开的, 同时还存在指针变化的操作, 因此边界可能比较难计算. 假如你比较懒, 就直接将数组大小/对答案贡献的范围调大一些, 这样可以省去不少麻烦.

#include <cstdio>
#include <cctype>
#include <vector>
#include <algorithm>
#include <cstring> namespace Zeonfai
{
inline int getInt()
{
int a = 0, sgn = 1;
char c;
while(! isdigit(c = getchar())) if(c == '-') sgn *= -1;
while(isdigit(c)) a = a * 10 + c - '0', c = getchar();
return a * sgn;
}
}
const int N = (int)5e4;
int n;
struct result
{
long long *first, *second;
inline result() {}
inline result(long long *_first, long long *_second)
{
first = _first; second = _second;
}
};
long long ans;
struct tree
{
struct node
{
std::vector<node*> edg;
int maxDepth, dep;
node *hvy;
inline void clear()
{
edg.clear(); hvy = NULL;
}
}nd[N + 1];
inline void clear()
{
for(int i = 1; i <= n; ++ i) nd[i].clear();
}
inline void addEdge(int u, int v)
{
nd[u].edg.push_back(nd + v); nd[v].edg.push_back(nd + u);
}
void getDepth(node *u, node *pre)
{
u->maxDepth = u->dep = pre == NULL ? 0 : pre->dep + 1;
for(auto v : u->edg) if(v != pre)
{
getDepth(v, u); u->maxDepth = std::max(u->maxDepth, v->maxDepth);
if(u->hvy == NULL || v->maxDepth > u->hvy->maxDepth) u->hvy = v;
}
}
result decomposition(node *u, node *pre, node *tp)
{
result res;
long long *f, *g;
if(u->hvy != NULL) res = decomposition(u->hvy, u, tp), f = res.first - 1, g = res.second + 1;
else
{
int len = u->dep - tp->dep + 10; //懒得想边界了, 直接开大一些, 求对答案的时候也求多一些就可以了
f = new long long[len << 1]; memset(f, 0, len << 1 << 3); f += len;
g = new long long[len << 1]; memset(g, 0, len << 1 << 3);
}
f[0] = 1; ans += g[0];
for(auto v : u->edg) if(v != pre && v != u->hvy)
{
int len = v->maxDepth - v->dep + 1;
res = decomposition(v, u, v); long long *_f = res.first, *_g = res.second;
for(int i = 1; i <= len; ++ i) ans += g[i] * _f[i - 1] + _g[i] * f[i - 1];
for(int i = 1; i <= len; ++ i) g[i] += _f[i - 1] * f[i];
for(int i = 0; i <= len; ++ i) g[i] += _g[i + 1];
for(int i = 1; i <= len; ++ i) f[i] += _f[i - 1];
}
return result(f, g);
}
}T;
int main()
{ #ifndef ONLINE_JUDGE freopen("thr.in", "r", stdin);
freopen("thr.out", "w", stdout); #endif using namespace Zeonfai;
while(n = getInt())
{
T.clear();
for(int i = 1; i < n; ++ i)
{
int u = getInt(), v = getInt();
T.addEdge(u, v);
}
T.getDepth(T.nd + 1, NULL);
ans = 0;
T.decomposition(T.nd + 1, NULL, T.nd + 1);
printf("%lld\n", ans);
}
}

最新文章

  1. mybatis多表连接在一起查询
  2. 如何遍历HashMap
  3. Bootstrap系列 -- 41. 带表单的导航条
  4. ubuntu14 谷歌输入法
  5. 关于windows中的快捷键
  6. 如何修改linux系统主机名称
  7. oracle 11g 64位安装32位客户端和PL/SQL
  8. java单元测试(Junit)
  9. Bootstrap导航悬浮顶部,stickUp
  10. 几种访问其他域swf文件,或本地浏览器运行环境【安全沙箱】冲突解决方法
  11. POJ 1062 昂贵的聘礼详解最短路变形
  12. hibernate系列笔记(2)---Hibernate的核心API
  13. 数据库安全性操作——操作原则及SQL注入
  14. JVM学习笔记二:垃圾收集算法
  15. Java EE 之 过滤器入门学习与总结(1)
  16. java中mysql查询报错java.sql.SQLException: Before start of result set
  17. Android自动化测试学习路线
  18. codeforces 798 D. Mike and distribution
  19. jQuery插件实例五:手风琴效果[动画效果可配置版]
  20. JavaScript 消息框,警告框,确认框,提示框

热门文章

  1. Java集合中的细节问题
  2. Careercup - Microsoft面试题 - 5799446021406720
  3. Python框架之Django学习笔记(十一)
  4. c++ primer 6 练习题 (非复习题)
  5. Leetcode 576.出界的路劲数
  6. CentOS7 haproxy+keepalived实现高可用集群搭建
  7. MySql数据库 - 2.启动与关闭
  8. firewalld的防火墙
  9. 多IP指定出口IP地址 如何指定云服务器源IP?
  10. 【Luogu】P2403所驼门王的宝藏(强连通分量)