https://zybuluo.com/ysner/note/1246822

题面

有三棵树,建两条边让他们相连,最大化所有点对距离之和。

  • \(40pts\ n\leq1000\)
  • \(100pts\ n\leq10^5\)

解析

\(40pts\)算法

似乎是怎么暴力怎么搞。

先预处理出三棵树内所有点对的距离\(dis\)和每一个点到其他点的距离和\(sum\),然后枚举最后另两颗树接到哪棵树上。

接着举例推导两颗树相连时对答案新产生的贡献(考场上推了\(1h\)???)

设建的边两端点分别是\(u\)点,对应\(A\)树,大小为\(size[A]\);

\(v\)点,对应\(B\)树,大小为\(size[B]\)。hree Trees

边长为\(len\)。

则贡献为$$sum[u]size[B]+lensize[A]size[B]+sum[v]size[A]$$

(这是整道题的钥匙)

然后枚举另两棵树接在第三棵树的哪个点上,同时计算答案即可。

答案中要同时统计:(设\(A\),\(C\)两树接到\(B\)上)

  • \(A,B,C\)三树各自内部的点对距离和
  • \(A,B\)两树产生的点对距离和
  • \(B,C\)两树产生的点对距离和
  • \(A,C\)两树产生的点对距离和

都是直接套用上面的式子。

复杂度\(O(n^2)\)。

\(100pts\)算法

注意到\(O(n^2)\)求\(sum\)似乎很傻逼。

可以换种思路,两次\(O(n)\)的树形\(DP\)解决。

第一次统计子树大小\(sz\)和子树所有点到根结点的距离之hree Trees和\(sum\)。

设\(u\)为父结点,\(v\)为子结点。

则\(sz[u]+=sz[v],sum[u]+=sz[v]+sum[v]\)。

第二次应用换根思想。

想一想若根变为儿子,原有的\(sum\)会怎么变化。

首先到\(n-sz[v]\)个点的距离会全部\(+1\)。

其次,每次往\(v\)移,\(sum\)都需新加上\(u\)结点的其他儿子及子树的贡献(\(u\)子树外的贡献都被\(u\)点统计过了)。

若设父亲\(sum\)加上的值为\(x\),

则儿子\(sum\)加上的值为\(x+n-sz[v]+sum[u]-sum[v]-sz[v]\)

于是预处理复杂度降到\(O(n)\)。

接着发现枚举另两棵树接在第三棵树的哪个点上,是复杂度瓶颈。

但是,如设\(B\)树\(sum\)值最大的点为\(j\),那么可以猜出接一棵树在上面肯定不会不优(由那个式子决定)。

于是只要枚举另一棵树接在哪个结点上即可。

由于要现算\(len\),复杂度\(O(nlogn)\)。

注意下式子运算过程中是否乘爆。

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<queue>
#define re register
#define il inline
#define ll long long
#define max(a,b) ((a)>(b)?(a):(b))
#define min(a,b) ((a)<(b)?(a):(b))
#define fp(i,a,b) for(re int i=a;i<=b;i++)
#define fq(i,a,b) for(re int i=a;i>=b;i--)
using namespace std;
const int N=2e5+100,inf=1e9;
ll ans;
struct Tree
{
int h[N],cnt,sz[N],pos,son[N],top[N],f[N];
ll mx,tot[N],sum[N],dd,d[N],n;
il Tree(){memset(h,-1,sizeof(h));cnt=0;pos=1;}
struct Edge{int to,nxt;}e[N<<1];
il void add(re int u,re int v){e[++cnt]=(Edge){v,h[u]};h[u]=cnt;e[++cnt]=(Edge){u,h[v]};h[v]=cnt;}
il void dfs1(re int u,re int fa)
{
sz[u]=1;f[u]=fa;d[u]=d[fa]+1;
for(re int i=h[u];i+1;i=e[i].nxt)
{
re int v=e[i].to;
if(v==fa) continue;
dfs1(v,u);
sz[u]+=sz[v];
if(sz[v]>sz[son[u]]) son[u]=v;
sum[u]+=sz[v]+sum[v];
}
}
il void dfs2(re int u,re int up)
{
top[u]=up;
if(son[u]) dfs2(son[u],up);
for(re int i=h[u];i+1;i=e[i].nxt)
{
re int v=e[i].to;
if(v==f[u]||v==son[u]) continue;
dfs2(v,v);
}
}
il void dfs3(re int u,re int fa,re ll las)
{
tot[u]=sum[u]+las;dd+=tot[u];
if(tot[u]>mx) mx=tot[u],pos=u;
for(re int i=h[u];i+1;i=e[i].nxt)
{
re int v=e[i].to;
if(v==fa) continue;
dfs3(v,u,las+n-sz[v]+sum[u]-sum[v]-sz[v]);
}
}
il int LCA(re int u,re int v)
{
while(top[u]^top[v])
{
if(d[top[u]]<d[top[v]]) swap(u,v);
u=f[top[u]];
}
return d[u]<d[v]?u:v;
}
il ll Dis(re int u,re int v){return d[u]+d[v]-2*d[LCA(u,v)];}
}T[4];
il void calc(re int A,re int B,re int C)
{
re ll s=(T[A].dd+T[B].dd+T[C].dd)+(T[A].mx*T[B].n+T[A].n*T[B].n+T[B].mx*T[A].n)+(T[C].mx*T[B].n+T[B].n*T[C].n+0)+(T[A].mx*T[C].n+0+T[C].mx*T[A].n),ysn=0;
fp(i,1,T[B].n) ysn=max(ysn,T[B].tot[i]*T[C].n+(T[B].Dis(T[B].pos,i)+2)*T[A].n*T[C].n);
ans=max(ans,s+ysn);
}
il ll gi()
{
re ll x=0,t=1;
re char ch=getchar();
while(ch!='-'&&(ch<'0'||ch>'9')) ch=getchar();
if(ch=='-') t=-1,ch=getchar();
while(ch>='0'&&ch<='9') x=x*10+ch-48,ch=getchar();
return x*t;
}
int main()
{
T[1].n=gi();T[2].n=gi();T[3].n=gi();
fp(i,1,T[1].n-1) T[1].add(gi(),gi());
fp(i,1,T[2].n-1) T[2].add(gi(),gi());
fp(i,1,T[3].n-1) T[3].add(gi(),gi());
fp(i,1,3) T[i].dfs1(1,0),T[i].dfs2(1,1),T[i].dfs3(1,0,0),T[i].dd/=2;
fp(i,1,3)
fp(j,1,3)
{
if(i==j) continue;
calc(i,j,6-i-j);
}
printf("%lld\n",ans);
return 0;
}

[CF391E2]Three Trees的更多相关文章

  1. [C#] C# 知识回顾 - 表达式树 Expression Trees

    C# 知识回顾 - 表达式树 Expression Trees 目录 简介 Lambda 表达式创建表达式树 API 创建表达式树 解析表达式树 表达式树的永久性 编译表达式树 执行表达式树 修改表达 ...

  2. hdu2848 Visible Trees (容斥原理)

    题意: 给n*m个点(1 ≤ m, n ≤ 1e5),左下角的点为(1,1),右上角的点(n,m),一个人站在(0,0)看这些点.在一条直线上,只能看到最前面的一个点,后面的被档住看不到,求这个人能看 ...

  3. [LeetCode] Minimum Height Trees 最小高度树

    For a undirected graph with tree characteristics, we can choose any node as the root. The result gra ...

  4. [LeetCode] Unique Binary Search Trees 独一无二的二叉搜索树

    Given n, how many structurally unique BST's (binary search trees) that store values 1...n? For examp ...

  5. [LeetCode] Unique Binary Search Trees II 独一无二的二叉搜索树之二

    Given n, generate all structurally unique BST's (binary search trees) that store values 1...n. For e ...

  6. 2 Unique Binary Search Trees II_Leetcode

    Given n, generate all structurally unique BST's (binary search trees) that store values 1...n. For e ...

  7. Finger Trees: A Simple General-purpose Data Structure

    http://staff.city.ac.uk/~ross/papers/FingerTree.html Summary We present 2-3 finger trees, a function ...

  8. Christmas Trees, Promises和Event Emitters

    今天有同事问我下面这段代码是什么意思: var MyClass = function() { events.EventEmitter.call(this); // 这行是什么意思? }; util.i ...

  9. 【leetcode】Unique Binary Search Trees (#96)

    Given n, how many structurally unique BST's (binary search trees) that store values 1...n? For examp ...

随机推荐

  1. Python isdigit() 方法检测字符串是否只由数字组成

    Python isdigit() 方法检测字符串是否只由数字组成

  2. c++通过CMake实现debug开关

    刚学cmake,很多东西还不是很懂,不过今天刚刚实现了通过CMake控制debug的开关,兴奋之余记录一下. 背景介绍: 最近参与到了一个大的C++项目,很多代码已经非常成熟,我来添加一些辅助功能,但 ...

  3. 字符串、散列--P1598 垂直柱状图

    题目描述 写一个程序从输入文件中去读取四行大写字母(全都是大写的,每行不超过100个字符),然后用柱状图输出每个字符在输入文件中出现的次数.严格地按照输出样例来安排你的输出格式. 输入输出格式 输入格 ...

  4. 洛谷——P2590 [ZJOI2008]树的统计(树链剖分模板练手)

    P2590 [ZJOI2008]树的统计 I. CHANGE u t : 把结点u的权值改为t II. QMAX u v: 询问从点u到点v的路径上的节点的最大权值 III. QSUM u v: 询问 ...

  5. MFC 课程总结

    <基于MFC框架开发>马志国 1491989781 MFC课程的组成 1.1 MFC应用程序的组成部分.执行机制和执行流程(10.5天). 1.2 Windows平台上的数据库访问技术(1 ...

  6. 简述systemd的新特性及unit常见类型分析、使用systemd管理编译安装的nginx

    1. systemd新特性 并行处理(同时启动)所有服务. 基于依赖关系定义的服务控制逻辑 系统状态快照 按需激活进程,只有第一次被访问时才会真正启动: 2. systemd的常见unit类型 Ser ...

  7. 使用Postman Interceptor发送带cookie的请求一直loading的解决法案

    很多web网页开发人员都知道Postman限制由于chrome安全的限制,发不出带cookie和带有自定义头部标签的请求.想要发出由于chrome安全的限制,发不出带cookie和带有自定义头部标签的 ...

  8. [luoguP1417] 烹调方案(背包DP)

    传送门 By tinylic 如果没有b[i]这个属性的话就是明显的01背包问题. 现在考虑相邻的两个物品x,y.假设现在已经耗费p的时间,那么分别列出先做x,y的代价: a[x]-(p+c[x])* ...

  9. HDU - 3407 - String-Matching Automata

    先上题目: String-Matching Automata Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K ...

  10. js的jsonp

    window.ajaxJsonp=function(params) { params = params || {}; params.data = params.data || {}; var json ...