题意

给一颗树,删除一条边再加一条边,使它仍为一颗树且任意两点间的距离的最大值最小。

题目数据范围描述有问题,n为1或重建不能使任意两点距离最大值变小,可以输出任意答案。

分析

删除一条边后会使它变成两颗树,两棵树的直径的中点相连一定是使距离最小的

红色的边为删除重建的边

在树上dp维护每个子树的最大直径\(h[x]\),和去除这个子树后的树的最大直径\(t[x]\),u为x的父亲,删除u-x这条边并重建后的树的最大直径为

\[\max\{\frac{h[x]+1}{2}+\frac{t[x]+1}{2}+1,h[x],t[x]\}
\]

设\(g[u]\)为以\(u\)为根的子树中\(u\)能到达的最远距离

设\(p[u]\)为去除以\(u\)为根的子树后\(u\)能到达的最远距离

自底向上

x为u的孩子,\(mx1\),\(mx2\)分别为\(g[x]\)的最大值和次大值

  • \(g[u]=max(g[x]+1)\)
  • \(h[u]=max\{h[x],g[u],mx1+mx2+2\}​\)

自顶向下

k为x的兄弟,\(mx1​\),\(mx2​\)分别为\(g[k]\)的最大值和次大值

  • \(p[x]=max(p[u]+1,g[k]+2)​\)
  • \(t[x]=max \{p[u],h[k],p[u]+g[k]+1,mx1+mx2+2 \}​\)

然后bfs找重建的边

实现细节很多,我写的比较乱,建议自己根据dp式子模拟一下

Code

#include<bits/stdc++.h>
#define fi first
#define se second
#define bug cout<<"--------------"<<endl
using namespace std;
typedef long long ll;
const double PI=acos(-1.0);
const double eps=1e-6;
const int inf=1e9;
const ll llf=1e18;
const int mod=1e9+7;
const int maxn=3e5+10;
int n;
vector<int>f[maxn];
typedef pair<int,int> pii;
pii e[maxn];
int g[maxn],h[maxn],p[maxn],t[maxn];
int ans=inf;
pii ans1,ans2;
void dfs1(int u,int fa){
int mx1=-inf,mx2=-inf;
int len=f[u].size();
int po=len;
for(int i=0;i<len;i++){
int x=f[u][i];
if(x==fa){
continue;
}
dfs1(x,u);
g[u]=max(g[x]+1,g[u]);
if(g[x]>mx1){
mx2=mx1;
mx1=g[x];
}else if(g[x]>mx2){
mx2=g[x];
}
h[u]=max(h[u],h[x]);
}
h[u]=max(h[u],g[u]);
h[u]=max(mx1+mx2+2,h[u]);
}
int pre[maxn],suf[maxn];
int pr[maxn],sf[maxn];
void dfs2(int u,int fa){
int len=f[u].size();
vector<int>q;
q.push_back(0);
for(int i=0;i<len+5;i++) pre[i]=suf[i]=pr[i]=sf[i]=-inf;
for(int i=0;i<len;i++){
int x=f[u][i];
if(x!=fa) q.push_back(x);
}
len=q.size()-1;
for(int i=1;i<=len;i++){
int x=q[i];
t[x]=max(t[x],t[u]);
pre[i]=max(pre[i-1],g[x]);
pr[i]=max(pr[i-1],h[x]);
}
for(int i=len;i>=1;i--){
int x=q[i];
suf[i]=max(suf[i+1],g[x]);
sf[i]=max(sf[i+1],h[x]);
}
for(int i=1;i<=len;i++){
int x=q[i];
p[x]=max(p[x],p[u]+1);
t[x]=max(p[u],t[x]);
}
for(int i=2;i<=len;i++){
int x=q[i];
t[x]=max(p[u]+1+pre[i-1],t[x]);
t[x]=max(pr[i-1],t[x]);
p[x]=max(pre[i-1]+2,p[x]);
}
for(int i=1;i<len;i++){
int x=q[i];
t[x]=max(p[u]+1+suf[i+1],t[x]);
t[x]=max(sf[i+1],t[x]);
p[x]=max(suf[i+1]+2,p[x]);
}
for(int i=2;i<len;i++){
int x=q[i];
t[x]=max(t[x],pre[i-1]+suf[i+1]+2);
}
int mx1=-inf,mx2=-inf;
for(int i=1;i<=len;i++){
int x=q[i];
t[x]=max(mx1+mx2+2,t[x]);
if(g[x]>mx1){
mx2=mx1;
mx1=g[x];
}else if(g[x]>mx2){
mx2=g[x];
}
}
mx1=mx2=-inf;
for(int i=len;i>=1;i--){
int x=q[i];
t[x]=max(mx1+mx2+2,t[x]);
if(g[x]>mx1){
mx2=mx1;
mx1=g[x];
}else if(g[x]>mx2){
mx2=g[x];
}
}
for(int i=1;i<=len;i++){
int x=q[i];
int dis=max(max(t[x],h[x]),(t[x]+1)/2+(h[x]+1)/2+1);
if(dis<ans){
ans=dis;
ans1=pii(x,u);
}
}
for(int i=0;i<(int)f[u].size();i++){
int x=f[u][i];
if(x==fa) continue;
dfs2(x,u);
}
}
int pe[maxn],vis[maxn];
queue<int>q;
int bfs(int fa){
int ret=fa;
memset(vis,0,sizeof(vis));
memset(pe,0,sizeof(pe));
q.push(fa);
vis[fa]=1;
while(!q.empty()){
int u=q.front();
q.pop();
ret=u;
int len=f[u].size();
for(int i=0;i<len;i++){
if(!vis[f[u][i]]){
q.push(f[u][i]);
pe[f[u][i]]=u;
vis[f[u][i]]=1;
}
}
}
return ret;
}
int fq[maxn],tot;
void dfs(int u,int s){
if(u==0) return;
fq[++tot]=u;
dfs(pe[u],s);
}
int find(int x){
tot=0;
int s=bfs(x);
int t=bfs(s);
dfs(t,s);
return fq[(tot+1)/2];
}
void work(){
for(int i=1;i<=n;i++){
f[i].clear();
}
for(int i=1;i<n;i++){
int a=e[i].fi,b=e[i].se;
if(a==ans1.fi&&b==ans1.se) continue;
if(b==ans1.fi&&a==ans1.se) continue;
f[a].push_back(b);
f[b].push_back(a);
}
ans2.fi=find(ans1.fi);
ans2.se=find(ans1.se);
cout<<ans<<endl;
cout<<ans1.fi<<" "<<ans1.se<<endl;
cout<<ans2.fi<<" "<<ans2.se<<endl;
}
int main(){
ios::sync_with_stdio(false);
//freopen("in","r",stdin);
cin>>n;
for(int i=1,a,b;i<n;i++){
cin>>a>>b;
f[a].push_back(b);
f[b].push_back(a);
e[i]=pii(a,b);
}
dfs1(1,0);
dfs2(1,0);
work();
return 0;
}

BZOJ 2651 城市改建 树形DP+模拟?的更多相关文章

  1. BZOJ 1149 风铃(树形DP)

    题目描述的实际是一颗二叉树,对于每个结点,要么满叉,要么无叉. 对于一种无解的简单情况,我们搜一遍树找到最浅的叶子结点1和最深的叶子结点2,如果dep[1]<dep[2]-1,则显然无解. 所以 ...

  2. bzoj 1369: Gem 树形dp

    题目大意 给出一棵树,要求你为树上的结点标上权值,权值可以是任意的正整数 唯一的限制条件是相临的两个结点不能标上相同的权值,要求一种方案,使得整棵树的总价值最小.N<=10000 题解 我们可以 ...

  3. 【BZOJ 3090】 树形DP

    3090: Coci2009 [podjela] Description 有 N 个农民, 他们住在 N 个不同的村子里. 这 N 个村子形成一棵树.每个农民初始时获得 X 的钱.每一次操作, 一个农 ...

  4. bzoj 1131 简单树形dp

    思路:随便想想就能想出来啦把...  卡了我一个vector... #include<bits/stdc++.h> #define LL long long #define fi firs ...

  5. bzoj 4987 Tree —— 树形DP

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4987 其实就是在树上找有 k 个点的连通块(路径上的点都选是最优的),之间的边都走了两遍,只 ...

  6. BZOJ 4753 二分+树形DP

    思路: 先二分答案 f[x][j]表示在x的子树里选j个点 f[x][j+k]=max(f[x][j+k],f[x][j]+f[v[i]][k]); 初始化 x!=0 -> f[x][1]=p[ ...

  7. [BZOJ 4033] [HAOI2015] T1 【树形DP】

    题目链接:BZOJ - 4033 题目分析 使用树形DP,用 f[i][j] 表示在以 i 为根的子树,有 j 个黑点的最大权值. 这个权值指的是,这个子树内部的点对间距离的贡献,以及 i 和 Fat ...

  8. [BZOJ 1907] 树的路径覆盖 【树形DP】

    题目链接:BZOJ - 1907 题目分析 使用树形 DP,f[x][0] 表示以 x 为根的子树不能与 x 的父亲连接的最小路径数(即 x 是一个折线的拐点). f[x][1] 表示以 x 为根的子 ...

  9. bzoj 4871: [Shoi2017]摧毁“树状图” [树形DP]

    4871: [Shoi2017]摧毁"树状图" 题意:一颗无向树,选两条边不重复的路径,删去选择的点和路径剩下一些cc,求最多cc数. update 5.1 : 刚刚发现bzoj上 ...

随机推荐

  1. X86逆向13:向程序中插入Dll

    本章我们将学习Dll的注入技巧,我们将把一个动态链接库永久的插入到目标程序中,让程序在运行后直接执行这个Dll文件,这一章的内容也可以看作是第八课的加强篇,第八课中我们向程序中插入了一个弹窗,有木有发 ...

  2. 华为设备ACL与NAT技术

    ACL 访问控制列表(Access Control Lists),是应用在路由器(或三层交换机)接口上的指令列表,用来告诉路由器哪些数据可以接收,哪些数据是需要被拒绝的,ACL的定义是基于协议的,它适 ...

  3. 分库分布的几件小事(四)分库分表的id主键生成

    1.问题 其实这是分库分表之后你必然要面对的一个问题,就是id咋生成?因为要是分成多个表之后,每个表都是从1开始累加,那肯定不对啊,需要一个全局唯一的id来支持.所以这都是你实际生产环境中必须考虑的问 ...

  4. 上述代码在JavaScript事件处理中

    上述代码在JavaScript事件处理中很常见,主要设置为与旧版本的Internet Explorer(主要在IE9之前)兼容,因为旧版本的IE不支持标准的W3C事件处理规范. 此代码中的e表示事件对 ...

  5. JPanel实现滚动条

    之前一直用JScrollPane里面放一个JTextArea,就可以在文本框内实现滚动条. 但是最近做一个小demo,需要在JPanel中实现滚动条,就找了下资料,做好了,现在记录一下,防止以后再用到 ...

  6. rabbitmq 连接报错 An unexpected connection driver error occured

    转自:https://blog.csdn.net/zht741322694/article/details/82801873 在服务器上安装了一个RabbitMq,并新创建了一个用户授予了管理员角色, ...

  7. nps内网渗透利用

    0x00 前言 对比了比较多的代理工具,如ew流量不稳定,容易断:frsocks目前免杀,也是容易断.frp需要落地配置文件,不符合渗透的规则.reg正向的socks,速度比较慢,扫描是个问题.其实我 ...

  8. odoo标识符

    class Book(models.Model): _name = "library.book" _description = "Book" _order = ...

  9. spket IDE插件更新地址

    http://www.agpad.com/update spket  IDE插件更新地址

  10. Rectangle Puzzle CodeForces - 281C (几何)

    You are given two rectangles on a plane. The centers of both rectangles are located in the origin of ...