Medium
You are given two integer arrays, source
and target
, both of length n
. You are also given an array allowedSwaps
where each allowedSwaps[i] = [ai, bi]
indicates that you are allowed to swap the elements at index ai
and index bi
(0-indexed) of array source
. Note that you can swap elements at a specific pair of indices multiple times and in any order.
The Hamming distance of two arrays of the same length, source
and target
, is the number of positions where the elements are different. Formally, it is the number of indices i
for 0 <= i <= n-1
where source[i] != target[i]
(0-indexed).
Return the minimum Hamming distance of source
and target
after performing any amount of swap operations on array source
.
Example 1:
Input: source = [1,2,3,4], target = [2,1,4,5], allowedSwaps = [[0,1],[2,3]]
Output: 1
Explanation: source can be transformed the following way:
Swap indices 0 and 1: source = [2,1,3,4]
Swap indices 2 and 3: source = [2,1,4,3]
The Hamming distance of source and target is 1 as they differ in 1 position: index 3.
Example 2:
Input: source = [1,2,3,4], target = [1,3,2,4], allowedSwaps = []
Output: 2
Explanation: There are no allowed swaps. The Hamming distance of source and target is 2 as they differ in 2 positions: index 1 and index 2.
Example 3:
Input: source = [5,1,2,4,3], target = [1,5,4,2,3], allowedSwaps = [[0,4],[4,2],[1,3],[1,4]]
Output: 0
Constraints:
n == source.length == target.length
1 <= n <= 105
1 <= source[i], target[i] <= 105
0 <= allowedSwaps.length <= 105
allowedSwaps[i].length == 2
0 <= ai, bi <= n - 1
ai != bi
import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;
import java.util.Map;
public class Solution {
public int minimumHammingDistance(int[] source, int[] target, int[][] allowedSwaps) {
int i;
int n = source.length;
int weight = 0;
int[] parent = new int[n];
for (i = 0; i < n; i++) {
parent[i] = i;
}
for (int[] swap : allowedSwaps) {
union(swap[0], swap[1], parent);
}
HashMap<Integer, List<Integer>> components = new HashMap<>();
for (i = 0; i < n; i++) {
find(i, parent);
List<Integer> list = components.getOrDefault(parent[i], new ArrayList<>());
list.add(i);
components.put(parent[i], list);
}
for (Map.Entry<Integer, List<Integer>> entry : components.entrySet()) {
weight += getHammingDistance(source, target, entry.getValue());
}
return weight;
}
private int getHammingDistance(int[] source, int[] target, List<Integer> indices) {
HashMap<Integer, Integer> list1 = new HashMap<>();
HashMap<Integer, Integer> list2 = new HashMap<>();
for (int i : indices) {
list1.put(target[i], 1 + list1.getOrDefault(target[i], 0));
list2.put(source[i], 1 + list2.getOrDefault(source[i], 0));
}
int size = indices.size();
for (Map.Entry<Integer, Integer> entry : list1.entrySet()) {
size -= Math.min(entry.getValue(), list2.getOrDefault(entry.getKey(), 0));
}
return size;
}
private void union(int x, int y, int[] parent) {
if (x != y) {
int a = find(x, parent);
int b = find(y, parent);
if (a != b) {
parent[a] = b;
}
}
}
private int find(int x, int[] parent) {
int y = x;
while (y != parent[y]) {
y = parent[y];
}
parent[x] = y;
return y;
}
}