POJ3585

POJ3585

题意

给定一棵 n n n个节点的无向带权树,要你找出一个节点作为根,向叶子节点流水,使流水量最大。

思路

换根DP
这种题往往都是先从某个点出发 D F S 1 DFS_1 DFS1,求出以这个节点 t t t为根的答案,并记录与答案相关的数据到数组中。
然后 D F S 2 DFS_2 DFS2 t t t节点出发,记录换根的答案,其中所有根的最大值就为所求值。
难点在于该如何去找寻其中的递推关系式。

d d d数组表示该点的最大水流量, f f f数组表示一该点为根的答案, d e g deg deg为度数。


当节点由 f a fa fa转移到 x x x时, f [ x ] f[x] f[x]分为两个部分,其中一个部分是 d [ x ] d[x] d[x],另一个部分为与 f a fa fa节点相连的部分。
注意:要对fa节点是否为根节点进行讨论。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int,int>P;
const double eps = 1e-8;
const int NINF = 0xc0c0c0c0;
const int INF  = 0x3f3f3f3f;
const ll  mod  = 1e9 + 7;
const ll  maxn = 1e6 + 5;
const int N = 2e5 + 5;

int n;
ll d[N],f[N],ans;
vector<pair<int,int> > G[N];

void dp(int x,int fa){
	for(auto c:G[x]){
		ll u=c.first,v=c.second;
		if(u==fa) continue;
		dp(u,x);
		if(d[u]!=0) d[x]+=min(d[u],v);
		else d[x]+=v;
	}
}

void dfs(int x,int fa,ll v){
	if(fa!=-1){
		f[x]=d[x];
		if(G[fa].size()!=1) f[x]+=min(f[fa]-min(d[x],v), v);
		else f[x]+=v;
	}
	for(auto c:G[x]){
		ll u=c.first,p=c.second;
		if(u==fa) continue;
		dfs(u,x,p);
	}
}

void solve(){
	cin>>n;
	for(int i=1;i<=n;i++) G[i].clear(),d[i]=0,f[i]=0;
	for(int i=1;i<n;i++){
		int u,v,w;cin>>u>>v>>w;
		G[u].push_back({v,w});
		G[v].push_back({u,w});
	}
	dp(1,-1);
	f[1]=d[1];
	dfs(1,-1,G[1][0].second);
	cout<<*max_element(f+1,f+1+n)<<'\n';
}

int main(){
	ios::sync_with_stdio(false);
	cin.tie(0);
	int T;cin>>T;
	while(T--){
		solve();
	}
	return 0;
}
全部评论

相关推荐

代码不跑我跑_bug...:北大杀完9✌杀,9✌杀完鼠鼠杀
你最希望上岸的公司是?
点赞 评论 收藏
分享
白火同学:先说结论,准大三不是特别好找实习,boss沟通300+没有实习是很正常的情况。一是暑期实习时间太短了,二是在这么多准大四都找不到实习,从实习时间和掌握技术层面,企业会优先看他们。 再说简历,其实985本+准大三到这水平的简历也很优秀了,要说的话,项目经历可以再优化一下,可以基本围绕采取STAR原则,分为项目概述、技术架构、技术亮点、实现结果,再发给AI润色一下。 最后说操作,准大三的话,如果想找实习那就多投,不过现在也7月中旬了,时间上已经略晚了。如果7月底实在找不到,也可以多刷点算法,多学点技术,这实习也不至于一定得有,当然有更好。
点赞 评论 收藏
分享
08-14 16:56
门头沟学院 Java
点赞 评论 收藏
分享
评论
点赞
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务