代码源 - 树上路径1

树上路径1

给你一个 \(n(1≤n≤2000)\)个点的树。

给你 \(m(1≤m≤2000)\) 条树上的简单路径,每个路径有个权值 \(a_i(1≤a_i≤109)\)

要求选择一些路径,使得每个点至多在一条路径上,并且路径的权值和最大。

输入格式

第一行两个整数n, m。

接下来一行n−1个数,\(f_2,f_3,…,f_n\),其中\(f_i(1≤f_i<i)\)表示 \(i\) 的父亲节点。

接下来mm行每行三个整数,\(u,v,a_i(u≠v)\),表示一条路径和它的权值。

输出格式

一个整数,表示答案。

#include <bits/stdc++.h>
using namespace std;
#define endl "\n"

typedef long long ll;
typedef pair<int, int> PII;

const int N = 2e3 + 10;
const ll INF = 1ll << 60;
vector<vector<int> > G(N);
vector<array<int, 3> > path[N];
int n, m;
int p[N];
int depth[N];
ll f[N], sf[N];

void dfs(int u)
{
	for (auto v : G[u])
	{
		dfs(v);
		sf[u] += f[v];
	}
	f[u] = sf[u];
	for (auto pth : path[u])
	{
		ll tmp = sf[u];
		int x = pth[0];
		while(x != u)
		{
			tmp += sf[x] - f[x];
			x = p[x];
		}
		x = pth[1];
		while(x != u)
		{
			tmp += sf[x] - f[x];
			x = p[x];
		}
		tmp += pth[2];
		f[u] = max(f[u], tmp);
	}
}
int main()
{
	scanf("%d%d", &n, &m);
	for (int i = 2; i <= n; i++)
	{
		scanf("%d", &p[i]);
		G[p[i]].push_back(i);
		depth[i] = depth[p[i]] + 1;
	}
	for (int i = 1; i <= m; i++)
	{
		int u, v, a;
		scanf("%d%d%d", &u, &v, &a);
		int x = u, y = v;
		while(x != y)
		{
			if (depth[x] > depth[y]) x = p[x];
			else y = p[y];
		}
		path[x].push_back({u, v, a});
	}
	dfs(1);
	printf("%lld\n", f[1]);
    return 0;
}
posted @ 2022-11-02 13:33  hzy0227  阅读(155)  评论(0编辑  收藏  举报