题目测试的时候感觉题就很恶心,10道都不会,不过在正式比赛的时候还是有队伍犀利的做出了6道,真的很厉害
这个是最水的一题了,但是对我还是很难,看着题解,划拉半天才看明白。
代码优化后JOJ跑了0.57s。(上午刚看明白,下午比赛就出了)
师兄给的题解:
树状DP
因为树是连通的且为无向边,所以可以假定1为根,从结点1开始向各个子树DFS
这次DFS过程中要对每个结点记录几个量
node[i]记录以i为根的子树(包括i结点)的T[k] 和
sum[i]记录以i为根的子树(包括i结点)中,每个结点到根的路径长度*T[k]的和
所以第1遍DFS得到的sum[1]是以1为the most convenient location的答案,那么开始枚举以不同结点为the most convenient location的情况
对于下图
1
/ \
2 3
/
4
比如转移到2节点,则以2节点为the most convenient location 的答案是
Sum[1] – sum_node[2] * g[1][2] + ( sum_node[1] – sum_node[2]) * g[1][2] = Sum[2]
(g[1][2]表示1到2这条边的权值)
接着对于以4节点为the most convenient location 的答案是
Sum[2] – sum_node[4] * g[2][4] + ( sum_node[1] – sum_node[4]) * g[2][4] = Sum[4]
以此类推,这样子DFS一遍即可
#include <cstdio> #include <string.h> const int maxn=100005; struct Edge{ int v,next,w; }edge[2*maxn]; int cnt,n,head[maxn]; long long node[maxn],sum[maxn],sum_node[maxn]; bool vis[maxn]; void addedge(int u,int v,int w){ edge[cnt].v=v; edge[cnt].w=w; edge[cnt].next=head[u]; head[u]=cnt++; edge[cnt].v=u; edge[cnt].w=w; edge[cnt].next=head[v]; head[v]=cnt++; } void dfs (int u) { int v,i,t=head[u],w; vis[u]=true ; sum_node[u]=node[u]; for (;~t;t=edge[t].next) { v=edge[t].v; if(vis[v])continue; w=edge[t].w; dfs(v); sum_node[u]+=sum_node[v]; sum[0]+=w*sum_node[v]; } } void dfs2 (int u) { int v,t,w; vis[u]=true ; for (t=head[u]; ~t ; t=edge[t].next) { v=edge[t].v; if(vis[v])continue; w=edge[t].w; sum[v]=sum[u]+(sum_node[0]-(sum_node[v]<<1))*w; dfs2(v); } } inline void init () { memset (head , -1 ,sizeof(head)); memset (vis , 0 , sizeof(vis)); sum[0]=0; cnt=0; } long long sovle () { int i; dfs(0); memset (vis , false , sizeof(vis)); dfs2(0); long long min_num=sum[0]; for (i=0 ; i<n ; ++i) if(min_num>sum[i]) min_num=sum[i]; return min_num; } int main () { int i,u,v,w; while (~scanf("%d",&n)) { init (); for (i=0 ; i<n ; ++i) scanf("%d",node+i); for (i=1 ; i<n ; ++i) { scanf("%d%d%d",&u,&v,&w); u--;v--; addedge (u,v,w); } printf("%lld\n",sovle()); } return 0; }