UVALive 4614 Moving to Nuremberg(树形DP)

题目链接:点击打开链接

一开始没看数据范围傻傻的用O(n^2)的算法怼了一波于是就妥妥的吃了一发TLE。。so sad。。

按照这道题的规模需要使用O(n)的方法,那么很明显就是用树形dp了。

首先随便取个点拉成颗有根树,这里取的是点1。

首先,我们将要去的目的地称为【指定点】,设其所有权值和为P。

要求的值可以看做是Σ【所有指定点】(当前点到一个指定点的路径权值和*指定点权值*2),然后将所有点求一遍再找最小值。

那么,将这个问题放在已经拉好的有根树里,当前点要求的Σ就可以分为两个部分:

1. 指定点为自己的后代们的时候的Σ

2. 指定点不是自己的后代们的时候的Σ

第一个Σ记为dp1[i],且设根为j的子树的指定点权值和为p[i],则有

dp1[i]=Σ(dp1[son]+p[son]*edge[i][son]),其中edge[i][son]为点i到这个儿子的边的权值。

接着是求第二个Σ。记为dp2[i],则有

dp2[i]=(dp1[prt]+dp2[prt]-dp1[i])+(P-p[i])*edge[i][prt],其中prt为i的父亲。

可以这么理解,dp2[i]的来源就是从父亲开始的【外面】的结点往下遍历分配给点i的。

那么每个点的解就是dp1[i]+dp2[i]了,取最小值并遍历一遍取出所有最小值点即可。

代码如下:

#include<bits/stdc++.h>
using namespace std;

struct edge
{
	int u,v;
	int w;
	edge(int uu=0,int vv=0,int ww=0):u(uu),v(vv),w(ww){}
}; 

vector<edge> e[50005];
bool vis[50005];

long long ans[50005];
long long Dans[50005],Dp[50005];

long long a[50005];

long long Solve(int pos)
{
	vis[pos]=true;
	vector<long long> v,ret;
	long long Ret=0;
	for(int i=0;i<e[pos].size();i++)
	{
		int &p=e[pos][i].v;
		if(vis[p])continue;
		v.push_back(Solve(p));
		ret.push_back(ans[p]);
		ans[pos]+=(v[v.size()-1]*e[pos][i].w+ret[ret.size()-1]);
		Ret+=v[v.size()-1];
	}
	Ret+=a[pos];
	int cnt=0;
	for(int i=0;i<e[pos].size();i++)
	{
		int &p=e[pos][i].v;
		if(vis[p])continue;
		Dp[p]=Ret-v[cnt];
		Dans[p]=ans[pos]-(v[cnt]*e[pos][i].w+ret[cnt])+Dp[p]*e[pos][i].w;
		cnt++;
	}
	vis[pos]=false;
	return Ret;
}

void dfs(int pos)
{
	vis[pos]=true;
	ans[pos]+=Dans[pos];
	for(int i=0;i<e[pos].size();i++)
	{
		int &p=e[pos][i].v;
		if(vis[p])continue;
		Dans[p]+=Dans[pos];
		Dans[p]+=Dp[pos]*e[pos][i].w;
		Dp[p]+=Dp[pos];
		dfs(p);
	}
	vis[pos]=false;
}

int main()
{
	int t;
	scanf("%d",&t);
	while(t--)
	{
		int n;
		scanf("%d",&n);
		for(int i=1;i<=n;i++)
			e[i].clear();
		for(int i=1;i<n;i++)
		{
			int u,v,w;
			scanf("%d%d%d",&u,&v,&w);
			e[u].push_back(edge(u,v,w));
			e[v].push_back(edge(v,u,w));
		}
		int m;
		scanf("%d",&m);
		memset(ans,0,sizeof(ans));
		memset(a,0,sizeof(a));
		memset(Dp,0,sizeof(Dp));
		memset(Dans,0,sizeof(Dans));
		for(int i=0;i<m;i++)
		{
			int p,w;
			scanf("%d%d",&p,&w);
			a[p]=w;
		}
		Solve(1);
		dfs(1);
//		for(int i=1;i<=n;i++)
//			printf("%lld %lld %lld\n",ans[i],Dans[i],Dp[i]);
		int mx=1;
		for(int i=2;i<=n;i++)
			if(ans[mx]>ans[i])mx=i;
		printf("%lld\n",ans[mx]*2);
		bool flag=false;
		for(int i=1;i<=n;i++)
		{
			if(ans[i]>ans[mx])continue;
			if(flag)printf(" ");
			else flag=true;
			printf("%d",i);
		}
		printf("\n");
	}
	return 0;
}

【这次有return 0了!】

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
提供的源码资源涵盖了安卓应用、小程序、Python应用和Java应用等多个领域,每个领域都包含了丰富的实例和项目。这些源码都是基于各自平台的最新技术和标准编写,确保了在对应环境下能够无缝运行。同时,源码中配备了详细的注释和文档,帮助用户快速理解代码结构和实现逻辑。 适用人群: 这些源码资源特别适合大学生群体。无论你是计算机相关专业的学生,还是对其他领域编程感兴趣的学生,这些资源都能为你提供宝贵的学习和实践机会。通过学习和运行这些源码,你可以掌握各平台开发的基础知识,提升编程能力和项目实战经验。 使用场景及目标: 在学习阶段,你可以利用这些源码资源进行课程实践、课外项目或毕业设计。通过分析和运行源码,你将深入了解各平台开发的技术细节和最佳实践,逐步培养起自己的项目开发和问题解决能力。此外,在求职或创业过程中,具备跨平台开发能力的大学生将更具竞争力。 其他说明: 为了确保源码资源的可运行性和易用性,特别注意了以下几点:首先,每份源码都提供了详细的运行环境和依赖说明,确保用户能够轻松搭建起开发环境;其次,源码中的注释和文档都非常完善,方便用户快速上手和理解代码;最后,我会定期更新这些源码资源,以适应各平台技术的最新发展和市场需求。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值