[bzoj4326][NOIP2015]运输计划
Description
公元2044年,人类进入了宇宙纪元。
L国有n个星球,还有n-1条双向航道,每条航道建立在两个星球之间,这n-1条航道连通了L国的所有星球。
小P掌管一家物流公司,该公司有很多个运输计划,每个运输计划形如:有一艘物流飞船需要从$u_i$号星球沿最快的宇航路径飞行到$v_i$号星球去。显然,飞船驶过一条航道是需要时间的,对于航道i,任意飞船驶过它所花费的时间为$t_i$,并且任意两艘飞船之间不会产生任何干扰。
为了鼓励科技创新,L国国王同意小P的物流公司参与L国的航道建设,即允许小P把某一条航道改造成虫洞,飞船驶过虫洞不消耗时间。在虫洞的建设完成前小P的物流公司就预接了m个运输计划。在虫洞建设完成后,这m个运输计划会同时开始,所有飞船一起出发。当这m个运输计划都完成时,小P的物流公司的阶段性工作就完成了。如果小P可以自由选择将哪一条航道改造成虫洞,试求出小P的物流公司完成阶段性工作所需要的最短时间是多少?
Input
第一行包括两个正整数n,m,表示L国中星球的数量及小P公司预接的运输计划的数量,星球从1到n编号。接下来n-1行描述航道的建设情况,其中第i行包含三个整数$a_i,b_i,t_i$,表示第i条双向航道修建在$a_i$与$b_i$两个星球之间,任意飞船驶过它所花费的时间为$t_i$。接下来m行描述运输计划的情况,其中第i行包含两个正整数$u_i$和$v_i$,表示第i个运输计划是从$u_i$号星球飞往$v_i$号星球。
Output
输出文件只包含一个整数,表示小P的物流公司完成阶段性工作所需要的最短时间。
Sample Input
6 3
1 2 3
1 6 4
3 1 7
4 3 6
3 5 5
3 6
2 5
4 5
Sample Output
11
HINT
$N\leq300000$
Solution
这道题就是求在一条边的时间不算的情况下,求$u_i$到$v_i$的最大值的最小值.
显然二分答案ans,然后判断对于所有花费时间$>ans$的$(u_i,v_i)$的交是否为空.
如果不为空,若满足$max\{(u_i,v_i)\}-max\{e[i].w|i\in$所有花费时间$>ans$的$(u_i,v_i)$的交$\}\leq ans$,则合法.
求路径交是否为空=求是否存在一条边所有路径都经过.
$w[i]$表示$i$连向它父亲的那条边有多少条路径经过,
每次$w[u_i]+1,w[v_i]+1,w[lca[(u_i,v_i)]-2$,向上传递即可.
#include
#include
#include
#include
#include
#include
#include
#include
#include
#include
#define K 20
#define N 300005
#define M 600005
#define INF 300000000
using namespace std;
typedef long long ll;
struct graph{
int nxt,to,w;
}e[M];
struct line{
int l,r,f,w;
}a[N];
int f[N][K],g[N],t[N],w[N],ef[N],dep[N],n,m,mk,sum,cnt;
bool v[N],flag;
stack s;
inline int read(){
int ret=0;char c=getchar();
while(!isdigit(c))
c=getchar();
while(isdigit(c)){
ret=(ret<<1)+(ret<<3)+c-'0';
c=getchar();
}
return ret;
}
inline void addedge(int x,int y,int w){
e[++cnt].nxt=g[x];g[x]=cnt;e[cnt].to=y;e[cnt].w=w;
}
inline void dfs(int u){
dep[u]=1;s.push(u);
while(!s.empty()){
u=s.top();s.pop();
if(u==1) for(int i=0;i>=1)
if(h&1) x=f[x][i];
return x;
}
inline int lca(int x,int y){
int i,t;
if(dep[x]dep[u])
s.push(e[i].to);
}
else{
s.pop();
for(int i=g[u];i;i=e[i].nxt)
if(dep[e[i].to]>dep[u])
w[u]+=w[e[i].to];
if(w[u]==sum){
flag=true;
mk=max(mk,ef[u]);
}
}
}
}
inline bool chk(int k){
int ma=0;mk=sum=0;
memset(w,0,sizeof(w));
for(int i=1;i<=m;++i){
if(a[i].w>k){
++sum;ma=max(ma,a[i].w);
++w[a[i].l];++w[a[i].r];w[a[i].f]-=2;
}
}
flag=false;total(1);
return flag&&ma-mk<=k;
}
inline void init(){
n=read();m=read();
for(int i=1,j,k,l;i>1;
if(chk(mid)) r=mid;
else l=mid+1;
}
printf("%d\n",l);
}
int main(){
freopen("transport.in","r",stdin);
freopen("transport.out","w",stdout);
init();
fclose(stdin);
fclose(stdout);
return 0;
}