Solution

一个定理: 把两棵树用一条边练成一棵树后, 树的直径在原来两棵树的四个直径端点中产生.

放到这一题, 我们通过DP先求出大树中以每个点为根的子树中的直径, 再取每棵小树中与其他树有连边的点以及两个直径端点作为虚树上的关键点, 建虚树再求一次直径即可.


#include <cstdio>
#include <cctype>
#include <vector>
#include <map>
#include <algorithm>
#define vector std::vector
#define map std::map
#define sort std::sort
#define swap std::swap 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)3e5, M = (int)3e5, LOG = 19;
vector<int> ky[M + 1];
struct virtualTree
{
struct node;
struct edge
{
node *v; int w;
inline edge(node *_v, int _w) {v = _v; w = _w;}
};
struct node
{
vector<edge> edg;
long long dep, maxLen;
node *dmt[2];
inline node() {edg.clear();}
} *rt;
inline void addEdge(node *u, node *v, int len)
{
u->edg.push_back(edge(v, len)); v->edg.push_back(edge(u, len));
}
node* DFS(node *u, node *pre)
{
u->dmt[0] = u->dmt[1] = u;
for(auto edg : u->edg) if(edg.v != pre)
{
node *v = edg.v; v->dep = u->dep + edg.w;
node *res = DFS(v, u);
if(res->dep > u->dmt[0]->dep) u->dmt[1] = u->dmt[0], u->dmt[0] = res; else if(res->dep > u->dmt[1]->dep) u->dmt[1] = res;
}
u->maxLen = u->dmt[0]->dep + u->dmt[1]->dep - (u->dep << 1);
node *res = u->dmt[0];
for(auto edg : u->edg) if(edg.v != pre && edg.v->maxLen > u->maxLen) u->dmt[0] = edg.v->dmt[0], u->dmt[1] = edg.v->dmt[1], u->maxLen = edg.v->maxLen;
return res;
}
inline long long getAnswer()
{
rt->dep = 0;
DFS(rt, NULL);
return rt->maxLen;
}
}vir;
struct edge
{
int id[2], u[2];
}edg[M];
struct node
{
vector<node*> edg;
int dep, dfn, maxLen;
node *dmt[2], *anc[LOG];
map<int, virtualTree::node*> mp;
inline node() {edg.clear(); mp.clear();}
};
inline int cmp(node *a, node *b) {return a->dfn < b->dfn;}
struct tree
{
node nd[N + 1];
inline void addEdge(int u, int v) {nd[u].edg.push_back(nd + v); nd[v].edg.push_back(nd + u);}
int clk;
node* DFS(node *u, node *pre)
{
u->dfn = clk ++; u->dep = pre == NULL ? 0 : pre->dep + 1;
u->dmt[0] = u->dmt[1] = u;
u->anc[0] = pre;
for(int i = 1; i < LOG; ++ i) u->anc[i] = u->anc[i - 1] == NULL ? NULL : u->anc[i - 1]->anc[i - 1];
for(auto v : u->edg) if(v != pre)
{
node *res = DFS(v, u);
if(res->dep > u->dmt[0]->dep) u->dmt[1] = u->dmt[0], u->dmt[0] = res; else if(res->dep > u->dmt[1]->dep) u->dmt[1] = res;
}
u->maxLen = u->dmt[0]->dep + u->dmt[1]->dep - (u->dep << 1);
node *res = u->dmt[0];
for(auto v : u->edg) if(v != pre && v->maxLen > u->maxLen) u->dmt[0] = v->dmt[0], u->dmt[1] = v->dmt[1], u->maxLen = v->maxLen;
return res;
}
inline void pretreat()
{
clk = 0;
DFS(nd + 1, NULL);
}
inline node* getLCA(node *u, node *v)
{
if(u->dep < v->dep) swap(u, v);
for(int i = LOG - 1; ~ i; -- i) if(u->dep - (1 << i) >= v->dep) u = u->anc[i];
if(u == v) return u;
for(int i = LOG - 1; ~ i; -- i) if(u->anc[i] != v->anc[i]) u = u->anc[i], v = v->anc[i];
return u->anc[0];
}
inline void buildVirtualTree(int id)
{
static node* a[N]; int cnt = 0;
for(auto u : ky[id]) a[cnt ++] = nd + u;
sort(a, a + cnt, cmp);
static node *stk[N]; int tp = 0;
for(int i = 0; i < cnt; ++ i) if(! i || a[i] != a[i - 1])
{
node *u = a[i]; vir.rt = u->mp[id] = new virtualTree::node;
if(tp > 1)
{
node *LCA = getLCA(u, stk[tp - 1]);
if(LCA != stk[tp - 1])
{
while(tp >= 2 && stk[tp - 2]->dep >= LCA->dep) vir.addEdge(stk[tp - 2]->mp[id], stk[tp - 1]->mp[id], stk[tp - 1]->dep - stk[tp - 2]->dep), -- tp;
if(stk[tp - 1] != LCA)
{
LCA->mp[id] = new virtualTree::node;
vir.addEdge(LCA->mp[id], stk[tp - 1]->mp[id], stk[tp - 1]->dep - LCA->dep);
-- tp;
stk[tp ++] = LCA;
}
}
}
stk[tp ++] = u;
}
for(int i = 0; i < tp - 1; ++ i) vir.addEdge(stk[i]->mp[id], stk[i + 1]->mp[id], stk[i + 1]->dep - stk[i]->dep);
}
inline void addVirtualTreeEdge(int id)
{
vir.addEdge(nd[edg[id].u[0]].mp[edg[id].id[0]], nd[edg[id].u[1]].mp[edg[id].id[1]], 1);
}
}T;
int main()
{ #ifndef ONLINE_JUDGE freopen("diameter.in", "r", stdin);
freopen("diameter.out", "w", stdout); #endif using namespace Zeonfai;
int n = getInt(), m = getInt();
for(int i = 1; i < n; ++ i)
{
int u = getInt(), v = getInt();
T.addEdge(u, v);
}
T.pretreat();
static int rt[M + 1];
for(int i = 1; i <= m; ++ i) rt[i] = getInt(), ky[i].push_back(rt[i]), ky[i].push_back(T.nd[rt[i]].dmt[0] - T.nd), ky[i].push_back(T.nd[rt[i]].dmt[1] - T.nd);
for(int i = 1; i < m; ++ i) for(int j = 0; j < 2; ++ j) edg[i].id[j] = getInt(), edg[i].u[j] = getInt(), ky[edg[i].id[j]].push_back(edg[i].u[j]);
for(int i = 1; i <= m; ++ i) T.buildVirtualTree(i);
for(int i = 1; i < m; ++ i) T.addVirtualTreeEdge(i);
printf("%lld\n", vir.getAnswer() + 1);
}

2016北京集训测试赛(八)Problem C: 直径的更多相关文章

  1. 【2016北京集训测试赛(八)】 crash的数列 (思考题)

    Description 题解 题目说这是一个具有神奇特性的数列!这句话是非常有用的因为我们发现,如果套着这个数列的定义再从原数列引出一个新数列,它居然还是一样的...... 于是我们就想到了能不能用多 ...

  2. 2016北京集训测试赛(十七)Problem C: 数组

    Solution 线段树好题. 我们考虑用last[i]表示\(i\)这个位置的颜色的上一个出现位置. 考虑以一个位置\(R\)为右端点的区间最远能向左延伸到什么位置: \(L = \max_{i \ ...

  3. 2016北京集训测试赛(十七)Problem B: 银河战舰

    Solution 好题, 又是长链剖分2333 考虑怎么统计答案, 我场上的思路是统计以一个点作为结尾的最长上升链, 但这显然是很难处理的. 正解的方法是统计以每个点作为折弯点的最长上升链. 具体的内 ...

  4. 2016北京集训测试赛(十七)Problem A: crash的游戏

    Solution 相当于要你计算这样一个式子: \[ \sum_{x = 0}^m \left( \begin{array}{} m \\ x \end{array} \right) \left( \ ...

  5. 2016北京集训测试赛(十六)Problem C: ball

    Solution 这是一道好题. 考虑球体的体积是怎么计算的: 我们令\(f_k(r)\)表示\(x\)维单位球的体积, 则 \[ f_k(1) = \int_{-1}^1 f_{k - 1}(\sq ...

  6. 2016北京集训测试赛(十六)Problem B: river

    Solution 这题实际上并不是构造题, 而是一道网络流. 我们考虑题目要求的一条路径应该是什么样子的: 它是一个环, 并且满足每个点有且仅有一条出边, 一条入边, 同时这两条边的权值还必须不一样. ...

  7. 2016北京集训测试赛(十六)Problem A: 任务安排

    Solution 这道题告诉我们, 不能看着数据范围来推测正解的时间复杂度. 事实证明, 只要常数足够小, \(5 \times 10^6\)也是可以跑\(O(n \log n)\)算法的!!! 这道 ...

  8. BZOJ 4543 2016北京集训测试赛(二)Problem B: thr 既 长链剖分学习笔记

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

  9. 2016北京集训测试赛(十四)Problem B: 股神小D

    Solution 正解是一个\(\log\)的link-cut tree. 将一条边拆成两个事件, 按照事件排序, link-cut tree维护联通块大小即可. link-cut tree维护子树大 ...

随机推荐

  1. TCP/IP网络编程之多进程服务端(一)

    进程概念及应用 我们知道,监听套接字会有一个等待队列,里面存放着不同客户端的连接请求,如果有一百个客户端,每个客户端的请求处理是0.5s,第一个客户端当然不会不满,但第一百个客户端就会有相当大的意见了 ...

  2. 基础_String

    String str1="hello"; String str2="hello"; String str3="hello"; String ...

  3. loj2056 「TJOI / HEOI2016」序列

    当年我还没学cdq的时候在luogu上写过树套树的代码orzzz ref #include <algorithm> #include <iostream> #include & ...

  4. Python-S9-Day122-Python爬虫

    11 初识爬虫 11 初识爬虫 12 今日作业 11.1 初识黄页: 11.2 互联网就是一张大的蜘蛛网: 网络爬虫(又被称为网页蜘蛛,网络机器人,在FOAF社区中间,更经常的称为网页追逐者),是一种 ...

  5. python学习-- django 2.1.7 ajax 请求 进阶版

    #原来版本 $.get("/add/",{'a':a,'b':b}, function(ret){ $('#result').html(ret)}) #进阶版  $.get(&qu ...

  6. PHP 命名空间和自动加载

    PHP 命名空间 php5.3 之后引入了命名空间的特性,从本质上讲,命名空间就是一个容器,你可以将类.函数和变量放在其中,在命名空间中,你可以无条件地访问这些项,在命名空间之外,必须导入或引用命名空 ...

  7. redis命令monitor详解

    通过monitor这个命令可以查看数据库在当前做了什么操作,对于管理redis数据库有这很大的帮助 如图示,在redis客户端进行操作显示info,另一个窗口打开monitor就会显示出这个命令的操作 ...

  8. [问题解决]NotImplementedError 错误原因:子类没有实现父类要求一定要实现的接口

    NotImplementedError: 子类没有实现父类要求一定要实现的接口. 在面向对象编程中,父类中可以预留一个接口不实现,要求在子类中实现.如果一定要子类中实现该方法,可以使用raise No ...

  9. [问题解决]docker启动不了

    问题描述:昨天下午整合了同事的代码,发现docker启动好后,docker ps查看不到,docker ps -a发现docker容器没有启动. 尝试多次启动发现都是启动不了. 经过搜索发现 http ...

  10. MySql 存储过程实例(附完整注释)(转)

    MySql 存储过程实例(附完整注释) 将下面的语句复制粘贴可以一次性执行完,我已经测试过,没有问题! MySql存储过程简单实例:                                   ...