cf14D Two Paths(树的直径)
2024-10-19 11:07:14
题意:
N个点构成一棵树。树枝的长度都是1。
在当中找两条不相交【没有公共点】的路,使得二者长度之积最大。
(2 ≤ n ≤ 200)
思路:
一开始思路好麻烦,好麻烦,好麻烦,,,,,,,而且WA,,,,,
正解:
必定存在第三条路径连接两条最长路径。【因为是一棵树】。
去掉第三条路径上的某根树枝就可以将原树分成两个区域了,两条最长路径分别在一个区域里。
然后分别求两个区域的直径,相乘。
N不大,枚举。
代码:
int const N=210; int n;
vector<int> G[N];
int g[N][N];
queue<int> Q;
int vis[N]; void Input(){
cin>>n; mem(g,0);
rep(i,1,n) G[i].clear(); rep(i,1,n-1){
int a,b;
scanf("%d%d",&a,&b);
G[a].push_back(b);
G[b].push_back(a);
g[a][b]=1;
g[b][a]=1;
}
} int path(int x){ //从x出发,求直径
mem(vis,-1);
while(!Q.empty()) Q.pop();
Q.push(x);
vis[x]=0;
while(!Q.empty()){
int u=Q.front(); Q.pop();
int L=G[u].size();
rep(i,0,L-1){
int v=G[u][i];
if(g[u][v]==1 && vis[v]==-1){
vis[v]=vis[u]+1;
Q.push(v);
}
}
}
int ans=-1;
int vv=-1;
rep(i,1,n){
if(vis[i]>ans){
ans=vis[i];
vv=i;
}
}
while(!Q.empty()) Q.pop();
mem(vis,-1);
Q.push(vv);
vis[vv]=0;
while(!Q.empty()){
int u=Q.front(); Q.pop();
int L=G[u].size();
rep(i,0,L-1){
int v=G[u][i];
if(g[u][v]==1 && vis[v]==-1){
vis[v]=vis[u]+1;
Q.push(v);
}
}
}
ans=-1;
rep(i,1,n){
if(vis[i]>ans){
ans=vis[i];
}
}
return ans;
}
void Solve(){
int ans=0;
rep(i,1,n-1){
rep(j,i+1,n){
if(g[i][j]==1){
g[i][j]=g[j][i]=0;
int x1=path(i);
int x2=path(j);
ans=max(ans,x1*x2);
g[i][j]=g[j][i]=1;
}
}
}
printf("%d\n",ans);
} int main(){ Input();
Solve();
return 0;
}
最新文章
- Spark概述
- jquery设置自己的标识符
- RPM Version Comparison
- CF#335 Board Game
- IE8 浏览器自动保存文档副本,添加缓存
- 基于visual Studio2013解决面试题之0407数组差
- Func和Action的用法区别
- bootstrap file input 官方文档翻译
- 导入Mybatis_Spring项目遇到的问题
- 20, CSS 定义选择器
- Elasticsearch学习笔记(九)partial update
- 个人项目:Java实现WC
- 简易付弹窗问题FAQ
- Linux下的压缩和解压缩命令gzip/gunzip
- nginx的https代理http配置
- centos7升级内核
- UWP 大爆炸你个锤子
- Centos7安装最新版本的docker
- Eclipse color theme jsp javascript显示问题
- JAVA math包