BZOJ4326: NOIP2015 运输计划

题目大意:给出一棵带边权的树和m条路径,可以将一条边的边权变成0,求问最长的路径最短是多少。

题解:

暴力算法:将每条边变不变,用数据结构维护,更新答案。

这样显然过不掉。

通常最值问题考虑贪心和二分答案,这里我们使用二分答案,二分最长的路径是多少。

假如最长路径maxn<=当前二分的值mid,那么结果显然可行。

如果maxn-mid可以通过将一条边的边权变成0来消去,即这条边的边权>=maxn-mid,那么结果也可行。

现在问题转化成:路径长度>mid的所有路径中,是否存在一条公共边,满足这条公共边的边权>=maxn-mid。

如何判断公共边呢?我们记一下每条边经过的次数,如果经过次数等于路径数,则说明这条边是公共边。

经过次数用树上查分来维护就好。

可是你会发现你会被卡常。。。

在这里给出几个优化的方法:

1.快速读入。

2.缩小l和r的范围,由于只能将一条边的边权变成0,所以左边界是最长路径maxn-最大边权tmp,右边界就是最长路径maxn。

3.有很多时间是花费在求LCA上,预处理出每条路径两个端点的LCA。

当然还有更毒瘤的优化方式,这里就不一一赘述了。

#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<string>
#include<map>
#include<iostream>
#include<queue>
using namespace std;
#define isNum(a) (a>='0'&&a<='9')
#define SP putchar(' ')
#define EL putchar('\n')
#define File(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout)
template<class T1>void read(T1 &r_e_a_d);
template<class T1>void write(T1 w_r_i_t_e);
int n,m,len,x,y,z,from[300005],to[300005],head[300005];
struct EDGE{
	int to,next,num;
}edge[600005];
void add(int u,int v,int w){
	++len;
	edge[len].to=v;
	edge[len].next=head[u];
	edge[len].num=w;
	head[u]=len;
}
int fa[300005],dep[300005],son[300005],siz[300005],dist[300005];
void dfs1(int u,int father){
	fa[u]=father;dep[u]=dep[father]+1;siz[u]=1;
	for (register int i=head[u];i;i=edge[i].next){
		int v=edge[i].to;
		if (v!=father){
			dist[v]=dist[u]+edge[i].num;
			dfs1(v,u);
			siz[u]+=siz[v];
			if (son[u]==-1||siz[v]>siz[son[u]]) son[u]=v;
		}
	}
}
int top[300005];
void dfs2(int u,int tp){
	top[u]=tp;
	if (son[u]==-1) return ;
	dfs2(son[u],tp);
	for (register int i=head[u];i;i=edge[i].next){
		int v=edge[i].to;
		if (v!=fa[u]&&v!=son[u]){
			dfs2(v,v);
		}
	}
}
int LCA(int u,int v){
	while (top[u]!=top[v]){
		if (dep[top[u]]>dep[top[v]]) u=fa[top[u]];
		else v=fa[top[v]];
	}
	if (dep[u]<dep[v]) return u;
	return v;
}//树链剖分求LCA 
int tmp,maxn,l,r,mid,ans,cnt[300005],e,lca[300005],dis[300005],dif[300005];
void prepare(int u,int father){
	dif[u]=cnt[u];
	for (register int i=head[u];i;i=edge[i].next){
		int v=edge[i].to;
		if (v!=father){
			prepare(v,u);
			dif[u]+=dif[v];
		}
	}
}
bool dfs(int u,int father,int k,int ed){
	for (register int i=head[u];i;i=edge[i].next){
		int v=edge[i].to;
		if (v!=father){
			if (dif[v]==ed&&edge[i].num>=k) return 1;
			if (dfs(v,u,k,ed)) return 1;
		}
	}
	return 0;
}
bool check(int k){
	memset (dif,0,sizeof (dif));
	memset (cnt,0,sizeof (cnt));
	e=0;
	if (maxn<=k) return 1;
	for (register int i=1;i<=m;++i){
		if (dis[i]>k){
			++e;
			++cnt[from[i]];++cnt[to[i]];cnt[lca[i]]-=2;
		}
	}
	prepare(1,0);
	return dfs(1,0,maxn-k,e);
}//树上差分代码 
int main(){
	memset (son,-1,sizeof (son));
	read(n);read(m);
	for (register int i=1;i<n;++i){
		read(x);read(y);read(z);
		add(x,y,z);add(y,x,z);
		tmp=max(tmp,z);
	}
	dfs1(1,0);dfs2(1,1);
	for (register int i=1;i<=m;++i){
		read(from[i]);read(to[i]);
		lca[i]=LCA(from[i],to[i]);
		dis[i]=dist[from[i]]+dist[to[i]]-dist[lca[i]]*2;
		r=max(r,dis[i]);
	}
	maxn=r;l=r-tmp;
	while (l<=r){
		mid=l+r>>1;
		if (check(mid)) ans=mid,r=mid-1;
		else l=mid+1;
	}
	write(ans);EL; 	
	return 0;
}
template<class T1>void read(T1 &r_e_a_d){
    T1 k=0;
    char ch=getchar();
    int flag=1;
    while(!isNum(ch)){
        if(ch=='-'){
            flag=-1;
        }
        ch=getchar();
    }
    while(isNum(ch)){
        k=((k<<1)+(k<<3)+ch-'0');
        ch=getchar();
    }
    r_e_a_d=flag*k;
}
template<class T1>void write(T1 w_r_i_t_e){
    if(w_r_i_t_e<0){
        putchar('-');
        write(-w_r_i_t_e);
    }else{
        if(w_r_i_t_e<10){
            putchar(w_r_i_t_e+'0');
        }else{
            write(w_r_i_t_e/10);
            putchar((w_r_i_t_e%10)+'0');
        }
    }
}
 

  

转载于:https://www.cnblogs.com/DFTMR/p/10756238.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值