-
Notifications
You must be signed in to change notification settings - Fork 26
/
Copy path721. Accounts Merge.java
executable file
·185 lines (153 loc) · 6.67 KB
/
721. Accounts Merge.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
M
tags: DFS, Union Find, Hash Table
- space O(mn)
- time O(mn)
给一串account in format `[[name, email1, email2, email3], [name2, email,..]]`.
要求把所有account merge起来 (可能多个record记录了同一个人, by common email)
#### Method1: Union Find
- 构建 `Map<email, email parent>`, 然后再反向整合: parent -> list of email
- init with <email, email> for all emails
- 因为不同account可能串email, 那么把所有email union的时候, 不同account 的email也会被串起来
- 最终: 所有的email都被union起来, 指向一个各自union的 parent email
- UnionFind 的 parent map 可以反向输出所有 child under parent.
- 同时要维护一个 <email -> account name> 的map, 最终用来输出.
#### Method2: Hash Table solution, passed but very slow
- Definitely need iterate over accounts: merge them by email.
- Account object {name, list of email}
- map<email, account>
- 1. iterate over accounts
- 2. find if 'account' exist; if does, add emails
- 3. if not, add account to list and to map. map all emails to accounts.
- output -> all accounts, and sort emails
- space O(mn): m row, n = emails
- time O(mn)
### DFS
- TODO
```
/*
Given a list accounts, each element accounts[i] is a list of strings,
where the first element accounts[i][0] is a name,
and the rest of the elements are emails representing emails of the account.
Now, we would like to merge these accounts. Two accounts definitely belong to the same person
if there is some email that is common to both accounts.
Note that even if two accounts have the same name,
they may belong to different people as people could have the same name.
A person can have any number of accounts initially, but all of their accounts definitely have the same name.
After merging the accounts, return the accounts in the following format:
the first element of each account is the name, and the rest of the elements are emails in sorted order.
The accounts themselves can be returned in any order.
Example 1:
Input:
accounts = [["John", "[email protected]", "[email protected]"], ["John", "[email protected]"], ["John", "[email protected]", "[email protected]"], ["Mary", "[email protected]"]]
Output: [["John", '[email protected]', '[email protected]', '[email protected]'], ["John", "[email protected]"], ["Mary", "[email protected]"]]
Explanation:
The first and third John's are the same person as they have the common email "[email protected]".
The second John and Mary are different people as none of their email addresses are used by other accounts.
We could return these lists in any order, for example the answer [['Mary', '[email protected]'], ['John', '[email protected]'],
Note:
The length of accounts will be in the range [1, 1000].
The length of accounts[i] will be in the range [1, 10].
The length of accounts[i][j] will be in the range [1, 30].
*/
/*
Method1: Union Find
Approach similar to LintCode(Find the weak connected component in directed graph)
UnionFind: Map<currEmail, parentEmail>
1. Group account into union. ( don't forget to preserve email -> name mapping)
2. Group parent -> children
3. output
*/
class Solution {
Map<String, String> accountMap = new HashMap<>();
Map<String, String> parentMap = new HashMap<>();
public List<List<String>> accountsMerge(List<List<String>> accounts) {
List<List<String>> rst = new ArrayList<>();
unionFind(accounts);
for (List<String> account : accounts) {
for (int i = 1; i < account.size() - 1; i++) {
union(account.get(i), account.get(i + 1)); // email across acounts will be auto-merged/linked in unionFind
}
}
Map<String, List<String>> map = new HashMap<>();
for (String email : parentMap.keySet()) {
String parent = find(email);
map.putIfAbsent(parent, new ArrayList<>());
map.get(parent).add(email);
}
for (List<String> list: map.values()) {
Collections.sort(list);
list.add(0, accountMap.get(list.get(0)));
rst.add(list);
}
return rst;
}
private void unionFind(List<List<String>> accounts) {
for (List<String> account : accounts) {
String name = account.get(0);
for (int i = 1; i < account.size(); i++) {
String email = account.get(i);
accountMap.put(email, name); // email -> name mapping
parentMap.put(email, email); // union parent map
}
}
}
private String find(String email) {
String parent = parentMap.get(email);
if (parent.equals(parentMap.get(parent))) return parent;
return find(parent);
}
private void union(String a, String b) {
String parentA = find(a), parentB = find(b);
if (!parentA.equals(parentB)) parentMap.put(parentA, parentB);
}
}
// Method2: Hash Table
/*
- use Account {name, emailSet}, and map all emails to the same node
- check each account: if any email is found, pick up that node and add all emails to it
- at the end, sort emails and return
*/
class Solution {
class Account {
String name;
Set<String> emails;
public Account(String name, Set<String> emails) {
this.name = name;
this.emails = emails;
}
}
public List<List<String>> accountsMerge(List<List<String>> accounts) {
Map<String, Account> map = new HashMap<>();
Set<Account> mergedAccounts = new HashSet<>();
for (List<String> row : accounts) {
Account account = new Account(row.get(0), new HashSet<>());
row.remove(0);
for (String email : row) {
account.emails.add(email);
if (map.containsKey(email)) {
Account existingAccount = map.get(email);
mergedAccounts.remove(existingAccount);
account.emails.addAll(existingAccount.emails);
}
}
mergedAccounts.add(account);
for (String email: account.emails) map.put(email, account);
}
// output as needed
return convert(mergedAccounts);
}
private List<List<String>> convert(Set<Account> accounts) {
List<List<String>> rst = new ArrayList<>();
for (Account account : accounts) {
List<String> row = new ArrayList<>();
List<String> emails = new ArrayList<>(account.emails);
Collections.sort(emails);
row.add(account.name);
row.addAll(emails);
rst.add(row);
}
return rst;
}
}
```