Hard
You are given an undirected tree with n
nodes labeled from 0
to n - 1
, and rooted at node 0
. You are given a 2D integer array edges
of length n - 1
, where edges[i] = [ai, bi]
indicates that there is an edge between nodes ai
and bi
in the tree.
You are also given a 0-indexed integer array cost
of length n
, where cost[i]
is the cost assigned to the ith
node.
You need to place some coins on every node of the tree. The number of coins to be placed at node i
can be calculated as:
i
is less than 3
, place 1
coin.3
distinct nodes in the subtree of node i
. If this product is negative, place 0
coins.Return an array coin
of size n
such that coin[i]
is the number of coins placed at node i
.
Example 1:
Input: edges = [[0,1],[0,2],[0,3],[0,4],[0,5]], cost = [1,2,3,4,5,6]
Output: [120,1,1,1,1,1]
Explanation: For node 0 place 6 * 5 * 4 = 120 coins. All other nodes are leaves with subtree of size 1, place 1 coin on each of them.
Example 2:
Input: edges = [[0,1],[0,2],[1,3],[1,4],[1,5],[2,6],[2,7],[2,8]], cost = [1,4,2,3,5,7,8,-4,2]
Output: [280,140,32,1,1,1,1,1,1]
Explanation: The coins placed on each node are: - Place 8 * 7 * 5 = 280 coins on node 0. - Place 7 * 5 * 4 = 140 coins on node 1. - Place 8 * 2 * 2 = 32 coins on node 2. - All other nodes are leaves with subtree of size 1, place 1 coin on each of them.
Example 3:
Input: edges = [[0,1],[0,2]], cost = [1,2,-2]
Output: [0,1,1]
Explanation: Node 1 and 2 are leaves with subtree of size 1, place 1 coin on each of them. For node 0 the only possible product of cost is 2 * 1 * -2 = -4. Hence place 0 coins on node 0.
Constraints:
2 <= n <= 2 * 104
edges.length == n - 1
edges[i].length == 2
0 <= ai, bi < n
cost.length == n
1 <= |cost[i]| <= 104
edges
represents a valid tree.import java.util.ArrayList;
import java.util.Comparator;
import java.util.List;
import java.util.PriorityQueue;
public class Solution {
private long[] result;
public long[] placedCoins(int[][] edges, int[] cost) {
int n = cost.length;
List<List<Integer>> g = new ArrayList<>();
for (int i = 0; i < n; i++) {
g.add(new ArrayList<>());
}
for (int[] e : edges) {
g.get(e[0]).add(e[1]);
g.get(e[1]).add(e[0]);
}
result = new long[n];
dp(g, cost, 0, -1);
return result;
}
private static class PQX {
PriorityQueue<Integer> min;
PriorityQueue<Integer> max;
}
private PQX dp(List<List<Integer>> g, int[] cost, int i, int p) {
if (i >= g.size()) {
PQX pqx = new PQX();
pqx.max = new PriorityQueue<>((a, b) -> b - a);
pqx.min = new PriorityQueue<>(Comparator.comparingInt(a -> a));
return pqx;
}
List<Integer> next = g.get(i);
PriorityQueue<Integer> pq = new PriorityQueue<>((a, b) -> b - a);
PriorityQueue<Integer> pq2 = new PriorityQueue<>(Comparator.comparingInt(a -> a));
if (cost[i] > 0) {
pq.add(cost[i]);
} else {
pq2.add(cost[i]);
}
for (int ne : next) {
if (ne != p) {
PQX r = dp(g, cost, ne, i);
while (!r.min.isEmpty()) {
int a = r.min.poll();
pq2.add(a);
}
while (!r.max.isEmpty()) {
int a = r.max.poll();
pq.add(a);
}
}
}
if (pq.size() + pq2.size() < 3) {
result[i] = 1;
} else {
int a = !pq.isEmpty() ? pq.poll() : 0;
int b = !pq.isEmpty() ? pq.poll() : 0;
int c = !pq.isEmpty() ? pq.poll() : 0;
int aa = !pq2.isEmpty() ? pq2.poll() : 0;
int bb = !pq2.isEmpty() ? pq2.poll() : 0;
result[i] = Math.max(0, (long) a * b * c);
result[i] = Math.max(result[i], Math.max(0, (long) a * aa * bb));
pq = new PriorityQueue<>((x, y) -> y - x);
pq.add(a);
pq.add(b);
pq.add(c);
pq2 = new PriorityQueue<>(Comparator.comparingInt(x -> x));
pq2.add(aa);
pq2.add(bb);
}
PQX pqx = new PQX();
pqx.min = pq2;
pqx.max = pq;
return pqx;
}
}