牛牛的朋友送了她一棵节点数为
的糖果树(1号点为根节点),树上的每个糖果都有一个颜色。
牛牛吃糖果有一个习惯,她每次都会吃掉一整个子树的糖果,她喜欢与众不同,所以每次吃之前她都会把出现次数最多的颜色的糖果扔掉(可能有多个出现次数最多的颜色,此时要全部扔掉),她可以选择任意一棵子树吃掉,但她只能吃一次,因此他想知道她一次能吃到的所有糖果的颜色的异或和最大是多少(如果吃到了多个颜色相同的糖果,也需要做多次异或),你能帮帮她吗?
第一行一个整数
。表示树的节点数量。
第二行
个整数
,表示节点i的颜色。
接下来
行,每行两个整数
,
表示节点
和节点
之间有一条边。
输出仅有一行,一个整数表示她一次能吃到的糖果的颜色的异或和最大值。
4 1 2 3 4 1 2 2 3 2 4
0
四个节点颜色各不相同。吃掉任意子树都会在吃之前把所有糖果扔掉,因此结果为0。
4 1 1 2 2 1 2 2 3 2 4
1
吃掉以节点3或节点4为根的子树都只有一个节点,结果都为0,以1为根节点时颜色为1和颜色为2的节点数量都为2,因此结果也为0。吃掉以2为根的子树时节点3和节点4颜色都为2被删除,结果为节点2的颜色1。
import java.util.*;
// 注意类名必须为 Main, 不要有任何 package xxx 信息
public class Main {
// 构建节点的邻接表
static List<Integer>[] tree;
// 颜色数组
static int[] colors;
// tong[i][j]:i节点的子树中颜色j出现的次数
static int[][] tong;
static int ans = 0; // 异或和
static int maxColor = 0;
public static void main(String[] args) {
Scanner in = new Scanner(System.in);
// 节点个数
int n = in.nextInt();
tree = new ArrayList[n + 1];
colors = new int[n + 1];
tong = new int[n + 1][1001];
// 初始化邻接表
for (int i = 1; i <= n; i++) {
tree[i] = new ArrayList<>();
}
// 初始化颜色数组
for (int i = 1; i <= n; i++) {
colors[i] = in.nextInt();
maxColor = Math.max(maxColor, colors[i]);
}
// 读取树的边
for (int i = 1; i < n; i++) {
int x = in.nextInt();
int y = in.nextInt();
tree[x].add(y);
tree[y].add(x);
}
// 从根节点(1)开始dfs
dfs(1, 0);
System.out.print(ans);
in.close();
}
public static void dfs(int x, int pre) {
// 当前节点的颜色个数 +1
tong[x][colors[x]]++;
// 遍历以x为根节点的子树
for (int i : tree[x]) {
if (i == pre)
continue;
dfs(i, x);
// 合并子节点的统计信息
for (int j = 1; j <= maxColor; j++) {
tong[x][j] += tong[i][j];
}
}
// 颜色出现次数最大的
int max = 0;
for (int j = 1; j <= maxColor; j++) {
max = Math.max(max, tong[x][j]);
}
// 计算剩余颜色的异或和
int res = 0;
for (int k = 1; k <= maxColor; k++) {
if (tong[x][k] != max) {
// 颜色个数是偶数时异或值为0,只有个数是奇数时异或和不为0
if (tong[x][k] % 2 == 1) {
res ^= k;
}
}
}
ans = Math.max(ans, res);
}
}