http://www.tyvj.cn/p/4878道路修建

我想我经大神点拨后终于明白了。。。回学校再写吧

时间限制:1s

内存限制:256MB

【问题述】

A国是一个商业高度发达的国家。它包含了n座城市,每座城商业都很发达。但不幸的是,A国的交通并没有像其商业那么发达,它仅仅保证了任意两座城市之间有路径存在,而且只存在唯一的一条!

拥有雄厚经济实力的商人们决定集资修建一条路,但在修建方案上各个商人都希望新建成的道路对自己利益最大。最终他们决定造一条路,使得两个城市间所需经过道路的数量的最大值尽可能小。为此他们提出了很多修建方案,但他们并不知道每一方案新建道路后最远城市间的最大值为多少,他们有多种修建方案,你能告诉他们每一方案对应的最远城市间的最大值吗?

【输入】

输入文件名为road.in。

第一行两个数n,m(1<=n、m<=3,000),分别表示城市个数和方案个数

接下来n-1行,每行两个数x、y,表示有一条道路连接x号城市和y号城市

m下来m行,每行两个数a、b,表示一个修建方案对应的两个城市

【输出】

输出文件名为road.out。

对于每组数据输出一行,包含一个数,表示新建道路后,最远城市间所需经过的道路数量

【输入输出样例】

road.in

road.out

8 2

1 3

2 3

3 4

4 5

5 6

6 7

6 8

3 6

1 8

3

5 更正

【数据说明】

对于40%的数据,1<=n,m<=300;

对于另外20%的数据,数据呈一条链

对于100%的数据,1<=n,m<=3000

#include<iostream>
#include<cstdio>
#include<queue>
#include<vector>
#include<algorithm>
#include<cstring>
using namespace std;
const int N=;
int n,m;
int h[N],nex[N*],to[N*],cnt;
int f[N];
bool vis[N];
int dep[N],deep[N];
int max_son[N];
int maxn,root;
void Add(int x,int y)
{
to[++cnt]=y,nex[cnt]=h[x],h[x]=cnt;
return;
}
void dfs1(int x,int tot)
{
vis[x]=;
for(int i=h[x];i;i=nex[i])
if(!vis[to[i]])
dfs1(to[i],tot+);
if(tot>maxn) maxn=tot,root=x;
return ;
}
int dfs2(int x,int tot,int last)
{
vis[x]=;f[x]=last;dep[x]=tot;
int sum=;
for(int i=h[x];i;i=nex[i])
if(vis[to[i]])
sum=max(sum,dfs2(to[i],tot+,x));
sum=max(sum,tot);
maxn=max(maxn,sum);
max_son[x]=sum;
return sum;
}
void work(int u,int v)
{
int t,len,ans=;
if(dep[u]>=dep[v])
t=u;else
if(dep[v]>dep[u])
t=v;
int minn=min(dep[v],dep[u]);
ans=dep[f[t]];
ans=max(ans,(minn+));
cout<<ans<<endl;
return ;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=,x,y;i<n;i++)
{
scanf("%d%d",&x,&y);
Add(x,y);Add(y,x);
}
f[]=;
dfs1(,);
maxn=;
dfs2(root,,);
for(int i=,u,v;i<=m;i++)
{
scanf("%d%d",&u,&v);
work(u,v);
}
return ;
}
#include<iostream>
#include<cstdio>
#include<queue>
#include<vector>
#include<algorithm>
#include<cstring>
using namespace std;
const int N=;
int n,m;
int h[N],nex[N*],to[N*],cnt;
int f[N];
bool vis[N];
int dep[N],deep[N];
int max_son[N];
int maxn,root;
void Add(int x,int y)
{ to[++cnt]=y,nex[cnt]=h[x],h[x]=cnt;}
void dfs1(int x,int tot)
{
vis[x]=;
for(int i=h[x];i;i=nex[i])
if(!vis[to[i]])
dfs1(to[i],tot+);
if(tot>maxn) maxn=tot,root=x;
return ;
}
int dfs2(int x,int tot,int last)
{
vis[x]=;f[x]=last;dep[x]=tot;
int sum=;
for(int i=h[x];i;i=nex[i])
if(vis[to[i]])
sum=max(sum,dfs2(to[i],tot+,x));
sum=max(sum,tot);
maxn=max(maxn,sum);
max_son[x]=sum;
return sum;
}
void work(int u,int v)
{
int t,len,ans=,last;
if(dep[u]>=dep[v])
t=u;else
if(dep[v]>dep[u])
t=v;
int minn=min(dep[v],dep[u]);
/*
if(max_son[t]!=maxn)
{
printf("%d\n",maxn);
return;
}else
{
int ans;
ans=maxn-(dep[t]-minn)+1;
ans=max(ans,minn+1+(dep[t]-minn)/2);
printf("%d\n",ans);
return ;
}
*/
/*
len=minn+1;last=t;
ans=max_son[t]-(dep[t]-minn)+1;
while(dep[t]>len)
{
for(int i=h[t];i;i=nex[i])
if((to[i]!=f[t])&&(to[i]!=last))
{
ans=max(ans,max_son[to[i]]-(dep[t]-len));
}
last=t;t=f[t];len++;
}
*/
ans=dep[f[t]]; printf("%d\n",ans);
return ;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=,x,y;i<n;i++)
{
scanf("%d%d",&x,&y);
Add(x,y);Add(y,x);
}
f[]=;
dfs1(,);
maxn=;
dfs2(root,,);
for(int i=,u,v;i<=m;i++)
{
scanf("%d%d",&u,&v);
work(u,v);
}
return ;
}
#include<iostream>
#include<cstdio>
#include<queue>
#include<vector>
#include<algorithm>
#include<cstring>
using namespace std;
const int N=;
int n,m;
int h[N],nex[N*],to[N*],cnt;
int f[N];
bool vis[N];
int dep[N],deep[N];
int max_son[N];
int maxn,root;
void Add(int x,int y)
{
to[++cnt]=y,nex[cnt]=h[x],h[x]=cnt;
return;
}
void dfs1(int x,int tot)
{
vis[x]=;
for(int i=h[x];i;i=nex[i])
if(!vis[to[i]])
dfs1(to[i],tot+);
if(tot>maxn) maxn=tot,root=x;
return ;
}
int dfs2(int x,int tot,int last)
{
vis[x]=;f[x]=last;dep[x]=tot;
int sum=;
for(int i=h[x];i;i=nex[i])
if(vis[to[i]])
sum=max(sum,dfs2(to[i],tot+,x));
sum=max(sum,tot);
maxn=max(maxn,sum);
max_son[x]=sum;
return sum;
}
void work(int u,int v)
{
int t,len,ans=;
if(dep[u]>=dep[v])
t=u;else
if(dep[v]>dep[u])
t=v;
int minn=min(dep[v],dep[u]);
len=minn+;
ans=minn+;
ans=max(ans,max_son[t]-(dep[t]-minn-));
while(dep[t]>len)
{
ans=max(ans,max_son[t]-(dep[t]-len));
len++,t=f[t];
} return ;
}
int main()
{
scanf("%d%d",&n,&m);
for(int i=,x,y;i<n;i++)
{
scanf("%d%d",&x,&y);
Add(x,y);Add(y,x);
}
f[]=;
dfs1(,);
maxn=;
dfs2(root,,);
for(int i=,u,v;i<=m;i++)
{
scanf("%d%d",&u,&v);
work(u,v);
}
return ;
}

P4878 道路修建-美国的更多相关文章

  1. 【BZOJ-2435】道路修建 (树形DP?)DFS

    2435: [Noi2011]道路修建 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 3115  Solved: 1002[Submit][Statu ...

  2. 【bzoj2435】[NOI2011]道路修建

    题目描述 在 W 星球上有 n 个国家.为了各自国家的经济发展,他们决定在各个国家之间建设双向道路使得国家之间连通.但是每个国家的国王都很吝啬,他们只愿意修建恰好 n – 1条双向道路. 每条道路的修 ...

  3. 【NOI2011】道路修建 BFS

    [NOI2011]道路修建 Description 在 W 星球上有 n 个国家.为了各自国家的经济发展,他们决定在各个国家之间建设双向道路使得国家之间连通.但是每个国家的国王都很吝啬,他们只愿意修建 ...

  4. 【BZOJ】2435: [Noi2011]道路修建(树形dp)

    http://www.lydsy.com/JudgeOnline/problem.php?id=2435 我怎么感觉那么水.. 坑的是,dfs会爆...好吧..用bfs.. //upd:我的智商也是醉 ...

  5. bzoj 2435: [Noi2011]道路修建 树上 dp

    2435: [Noi2011]道路修建 Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeOnline/pr ...

  6. 2435: [Noi2011]道路修建 - BZOJ

    Description 在 W 星球上有 n 个国家.为了各自国家的经济发展,他们决定在各个国家之间建设双向道路使得国家之间连通.但是每个国家的国王都很吝啬,他们只愿意修建恰好 n – 1条双向道路. ...

  7. NOI2011道路修建

    2435: [Noi2011]道路修建 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1974  Solved: 550[Submit][Status ...

  8. BZOJ 2435: [Noi2011]道路修建( dfs )

    NOI的水题...直接一遍DFS即可 ------------------------------------------------------------------------- #includ ...

  9. 道路修建 2(自创题+题解)(From NOI2011)

    道路修建这道题想来各位不陌生(传送门在此——Bzoj2435),看了此题,一开始以为是最初各个点处于分散状态,然后做了一下,直到发现标程都有点问题,才发现原题是说本来各点已经处于连接完毕的状态(phi ...

随机推荐

  1. BZOJ4088: [Sdoi2015]立体图

    高一联赛之后不久写的.当时看到这题就感觉特别优美.那个时候啥都不会,就只会这种模拟题,还只会最暴力的方法.对于每个方向的灯,枚举每个位置,手动枚举所有遮挡效果,并在枚举位置过程中传递遮挡效果. con ...

  2. ACM学习历程—HDU5410 CRB and His Birthday(动态规划)

    Problem Description Today is CRB's birthday. His mom decided to buy many presents for her lovely son ...

  3. ACM学习历程—CodeForces 176B Word Cut(字符串匹配 && dp && 递推)

    Description Let's consider one interesting word game. In this game you should transform one word int ...

  4. AIX 7.1上安装Oracle11g

    1. 上传oracle 11g介质到AIX 我下载的介质是aix.ppc64_11gR2_database_1of2.zip和aix.ppc64_11gR2_database_2of2.zip, 执行 ...

  5. 转 对APK进行重签名

    1.      生成Android APK包签名证书1).     在doc中切换到jdk的bin目录cd C:\Program Files\Java\jdk1.6.0_18\bin2).     运 ...

  6. Asp.Net 无法获取IIS拾取目录的解决办法[译]

    Asp.Net 无法获取IIS拾取目录的解决办法 作者:Jason Doucette  [MCP] 翻译:彭远志 原文地址:Fixing the cannot get IIS pickup direc ...

  7. Coding-Job:从研发到生产的容器化融合实践

    大家好,我是来自 CODING 的全栈开发工程师,我有幸在 CODING 参与了 Coding-Job 这个容器化的编排平台的研发.大家对 CODING 可能比较了解, Coding.net 是一个一 ...

  8. Thinkphp3.2 下载文件的方法

    今天做一个功能,刚好遇到了一个要下载文件功能的需求,所以把这个基于thinkphp3.2的文件下载功能,描述一下大概的实现方法. 网上有人说用a链接的方法实现,但是这种方法并不安全.所以我们还是用官方 ...

  9. Consuming JSON Strings in SQL Server

    https://www.simple-talk.com/sql/t-sql-programming/consuming-json-strings-in-sql-server/ Consuming JS ...

  10. 天梯赛L3-001. 凑零钱(01背包记录物品)

    L3-001. 凑零钱 时间限制 200 ms 内存限制 65536 kB 代码长度限制 8000 B 判题程序 Standard 作者 陈越 韩梅梅喜欢满宇宙到处逛街.现在她逛到了一家火星店里,发现 ...