bzoj3242 [Noi2013]快餐店

Description

小T打算在城市C开设一家外送快餐店。送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近的地方。 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连接起来,不存在任何两条道路连接了相同的两个建筑。任意两个建筑之间至少存在一条由双向道路连接而成的路径。小T的快餐店可以开设在任一建筑中,也可以开设在任意一条道路的某个位置上(该位置与道路两端的建筑的距离不一定是整数)。 现给定城市C的地图(道路分布及其长度),请找出最佳的快餐店选址,输出其与最远的顾客之间的距离。

Input

第一行包含一个整数N,表示城市C中的建筑和道路数目。
接下来N行,每行3个整数,Ai,Bi,Li(1≤i≤N;Li>0),表示一条道路连接了建筑Ai与Bi,其长度为Li 。

Output

仅包含一个实数,四舍五入保留恰好一位小数,表示最佳快餐店选址距离最远用户的距离。
注意:你的结果必须恰好有一位小数,小数位数不正确不得分。

Sample Input

1 2 1
1 4 2
1 3 2
2 4 1

Sample Output

2.0

HINT

数据范围
对于 10%的数据,N<=80,Li=1; 
对于 30%的数据,N<=600,Li<=100; 
对于 60% 的数据,N<=2000,Li<=10^9; 
对于 100% 的数据,N<=10^5,Li<=10^9

正解:基环树直径+树形$dp$。

这应该是$NOI2013$最水的一道题吧。。不过我还是写了$3$个小时,因为漏了一个情况,结果还有$95$分。。

很显然,直接求直径是错的,样例$2$就已经告诉你了。那么我们可以发现,这题其实是要求断掉环上一条边的直径,且使得这个直径最小。

我们先在环上随便找一个点作为根,然后$dfs$。先把所有树的情况处理完,这样对于每个点,$f[x]$表示$x$到它子树的最长链,用树形$dp$先搞一下就行了。

然后考虑环上怎么做。枚举断哪条边,然后我们可以记一个前缀最大值和一个后缀最大值。$v1[x]$表示环顶到$x$逆时针,$f[x]+dis(rt,x)$的最大值($rt$即为环顶),$v2[x]$表示环顶到$x$顺时针,$f[x]+dis(rt,x)$的最大值。注意,两个$dis$是不一样的,一个是逆时针的$dis$,另一个是顺时针的。

这样,我们枚举断边,假设我们断掉的是$x->x+1$这条边,那么断这条边的直径就是$f[rt]+max(v1[x],v2[x+1])$和$v1[x]+v2[x+1]$的最大值(注意此时的$f[rt]$可能已经被别的链更新了,所以必须加上)。然后我们对于所有的最大值,取一个最小值即可。

然后这就是我开始的$95$分算法,还有一种情况没有考虑到。。

注意到上面的情况,我们可以发现它算的是环上两点距离和它们子树的最长链,但是这两点是位于断边两侧的,也就是说,断边同侧的两点同样有可能构成直径。

那么如何计算同侧的情况呢?我们设$v1[x]$为环顶到$x$逆时针,$f[x]-dis(rt,x)$的最大值,$v2[x]$为环顶到$x$顺时针,$f[x]-dis(rt,x)$的最大值。那么这时,枚举断边$i->i+1$,答案有可能由$f[x]+dis(rt,x)-v1[x-1]$和$f[x+1]+dis(rt,x+1)-v2[x+2]$的最大值构成,注意到多余部分的$dis$一加一减,正好被抵消了,所以我们是不会算错的。我们算出这两个值以及上面两个值,取一个最大值,再对所有的最大值取一个最小值,就是答案了。

  1 //It is made by wfj_2048~
  2 #include <algorithm>
  3 #include <iostream>
  4 #include <cstring>
  5 #include <cstdlib>
  6 #include <cstdio>
  7 #include <vector>
  8 #include <cmath>
  9 #include <queue>
 10 #include <stack>
 11 #include <map>
 12 #include <set>
 13 #define inf (1LL<<60)
 14 #define N (500010)
 15 #define il inline
 16 #define RG register
 17 #define ll long long
 18 #define File(s) freopen(s".in","r",stdin),freopen(s".out","w",stdout)
 19
 20 using namespace std;
 21
 22 struct edge{ int nt,to; ll dis; }g[2*N];
 23
 24 int head[N],fa[N],dfn[N],low[N],st[N],n,rt,num,cnt;
 25 ll dis[N],f[N],d[N],v1[N],v2[N],u1[N],u2[N],ans;
 26
 27 il int gi(){
 28     RG int x=0,q=1; RG char ch=getchar();
 29     while ((ch<‘0‘ || ch>‘9‘) && ch!=‘-‘) ch=getchar();
 30     if (ch==‘-‘) q=-1,ch=getchar();
 31     while (ch>=‘0‘ && ch<=‘9‘) x=x*10+ch-48,ch=getchar();
 32     return q*x;
 33 }
 34
 35 il void insert(RG int from,RG int to,RG ll dis){
 36     g[++num]=(edge){head[from],to,dis},head[from]=num; return;
 37 }
 38
 39 il void circle(RG int rt,RG int x,RG ll len){
 40     RG int top=0,tp=1; RG ll D=len,res=inf,res1,res2;
 41     for (RG int i=x;i!=rt;i=fa[i]) ++tp,D+=dis[i]; top=tp;
 42     for (RG int i=x;i!=rt;i=fa[i]) st[tp--]=i; st[1]=rt;
 43     for (RG int i=2;i<=top;++i){
 44     d[i]=d[i-1]+dis[st[i]];
 45     v1[i]=max(v1[i-1],f[st[i]]+d[i]);
 46     u1[i]=max(u1[i-1],f[st[i]]-d[i]);
 47     }
 48     for (RG int i=top;i>1;--i){
 49     v2[i]=max(v2[i+1],f[st[i]]+D-d[i]);
 50     u2[i]=max(u2[i+1],f[st[i]]-D+d[i]);
 51     }
 52     for (RG int i=2;i<top-1;++i){
 53     res1=max(v1[i]+v2[i+1],f[rt]+max(v1[i],v2[i+1]));
 54     res2=max(f[st[i]]+d[i]+u1[i-1],f[st[i+1]]+D-d[i+1]+u2[i+2]);
 55     res=min(res,max(res1,res2));
 56     }
 57     res=min(res,f[rt]+v1[top]),res=min(res,f[rt]+v2[2]),ans=max(ans,res); return;
 58 }
 59
 60 il void tarjan0(RG int x,RG int p){
 61     fa[x]=p,dfn[x]=low[x]=++cnt; RG int v;
 62     for (RG int i=head[x];i;i=g[i].nt){
 63     v=g[i].to; if (v==p) continue;
 64     if (!dfn[v]){
 65         dis[v]=g[i].dis,tarjan0(v,x);
 66         low[x]=min(low[x],low[v]);
 67     } else low[x]=min(low[x],dfn[v]);
 68     }
 69     for (RG int i=head[x];i;i=g[i].nt){
 70     v=g[i].to; if (v==p) continue;
 71     if (fa[v]!=x && dfn[x]<dfn[v]) rt=x;
 72     }
 73     return;
 74 }
 75
 76 il void tarjan(RG int x,RG int p){
 77     fa[x]=p,dfn[x]=low[x]=++cnt; RG int v;
 78     for (RG int i=head[x];i;i=g[i].nt){
 79     v=g[i].to; if (v==p) continue;
 80     if (!dfn[v]){
 81         dis[v]=g[i].dis,tarjan(v,x);
 82         low[x]=min(low[x],low[v]);
 83     } else low[x]=min(low[x],dfn[v]);
 84     if (dfn[x]<low[v]){
 85         ans=max(ans,f[x]+f[v]+g[i].dis);
 86         f[x]=max(f[x],f[v]+g[i].dis);
 87     }
 88     }
 89     for (RG int i=head[x];i;i=g[i].nt){
 90     v=g[i].to; if (v==p) continue;
 91     if (fa[v]!=x && dfn[x]<dfn[v]) circle(x,v,g[i].dis);
 92     }
 93     return;
 94 }
 95
 96 il void work(){
 97     n=gi();
 98     for (RG int i=1,u,v,w;i<=n;++i) u=gi(),v=gi(),w=gi(),insert(u,v,w),insert(v,u,w);
 99     tarjan0(1,0); for (RG int i=1;i<=n;++i) dfn[i]=low[i]=fa[i]=dis[i]=0;
100     cnt=0,tarjan(rt,0); printf("%0.1Lf",0.5*(long double)ans); return;
101 }
102
103 int main(){
104     File("foodshop");
105     work();
106     return 0;
107 }
时间: 2025-01-03 05:53:40

bzoj3242 [Noi2013]快餐店的相关文章

BZOJ3242 [Noi2013]快餐店/UOJ126

本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作. 本文作者:ljh2000 作者博客:http://www.cnblogs.com/ljh2000-jump/转载请注明出处,侵权必究,保留最终解释权! Description 小T打算在城市C开设一家外送快餐店.送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近的地方. 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连

[noi2013]快餐店 基环树dp,单调队列维护最大值和次大值

#include<iostream> #include<cstdio> #include<cstring> #include<algorithm> using namespace std; #define N 220000 #define inf 0x3ffffffffffffffLL typedef long long ll; int v[N],e[N],ne[N],nn,w[N]; void add(int x,int y,int z){ ne[++nn

bzoj 3242: [Noi2013]快餐店 章鱼图

3242: [Noi2013]快餐店 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 266  Solved: 140[Submit][Status] Description 小 T打算在城市C开设一家外送快餐店.送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近 的地方. 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连接起来,不存在任何两条道路连接了相同的两

3242: [Noi2013]快餐店 - BZOJ

Description 小T打算在城市C开设一家外送快餐店.送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近的地方. 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连接起来,不存在任何两条道路连接了相同的两个建筑.任意两个建筑之间至少存在一条由双向道路连接而成的路径.小T的快餐店可以开设在任一建筑中,也可以开设在任意一条道路的某个位置上(该位置与道路两端的建筑的距离不一定是整数). 现给定城市C的地图(道路

[NOI2013]快餐店

题目描述 小T打算在城市C开设一家外送快餐店.送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近的地方. 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连接起来,不存在任何两条道路连接了相同的两个建筑.任意两个建筑之间至少存在一条由双向道路连接而成的路径.小T的快餐店可以开设在任一建筑中,也可以开设在任意一条道路的某个位置上(该位置与道路两端的建筑的距离不一定是整数). 现给定城市C的地图(道路分布及其长度)

动态规划:NOI2013 快餐店

Description 小 T打算在城市C开设一家外送快餐店.送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近 的地方. 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连接起来,不存在任何两条道路连接了相同的两个建筑.任意两个建筑之间至少存在一条由双向道路连接而成的路径.小T的快餐店可以开设在任一建筑 中,也可以开设在任意一条道路的某个位置上(该位置与道路两端的建筑的距离不一定是整数). 现给定城市C的地图

bzoj 3242: [Noi2013]快餐店

Description 小T打算在城市C开设一家外送快餐店.送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近的地方. 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连接起来,不存在任何两条道路连接了相同的两个建筑.任意两个建筑之间至少存在一条由双向道路连接而成的路径.小T的快餐店可以开设在任一建筑中,也可以开设在任意一条道路的某个位置上(该位置与道路两端的建筑的距离不一定是整数). 现给定城市C的地图(道路

「总结」$pdf$课: $tree$

一堆树.. 1.Codechef CUTTREE 有序点对\((x,y)\)如果联通,就对答案造成1的贡献. 只需要求出第\(i\)天期望联通的点对的个数. \((x,y)\)对第\(i\)天的贡献就是:\(\frac{(n-1-dis(x,y))^{\underline{i}}}{(n-1)^{\underline{i}}}\) 那么有: \[ans_i=\sum\limits_{d=0}^{n-1}\frac{(n-1-d)!(n-1-i)!}{(n-1-d-i)!(n-1)!}c_d \]

【BZOJ3242】【UOJ#126】【NOI2013】快餐店

NOI都是这种难度的题怎么玩嘛QAQ 原题: 小T打算在城市C开设一家外送快餐店.送餐到某一个地点的时间与外卖店到该地点之间最短路径长度是成正比的,小T希望快餐店的地址选在离最远的顾客距离最近的地方. 快餐店的顾客分布在城市C的N 个建筑中,这N 个建筑通过恰好N 条双向道路连接起来,不存在任何两条道路连接了相同的两个建筑.任意两个建筑之间至少存在一条由双向道路连接而成的路径.小T的快餐店可以开设在任一建筑中,也可以开设在任意一条道路的某个位置上(该位置与道路两端的建筑的距离不一定是整数). 现