Skip to content

Commit 5838e46

Browse files
committed
feat: add solutions to lcof2 problem: No.117
1 parent b675cac commit 5838e46

File tree

10 files changed

+378
-31
lines changed

10 files changed

+378
-31
lines changed

lcof2/剑指 Offer II 117. 相似的字符串/README.md

+212-1
Original file line numberDiff line numberDiff line change
@@ -50,22 +50,233 @@
5050

5151
<!-- 这里可写通用的实现逻辑 -->
5252

53+
并查集模板题。
54+
55+
模板 1——朴素并查集:
56+
57+
```python
58+
# 初始化,p存储每个点的父节点
59+
p = list(range(n))
60+
61+
# 返回x的祖宗节点
62+
def find(x):
63+
if p[x] != x:
64+
# 路径压缩
65+
p[x] = find(p[x])
66+
return p[x]
67+
68+
# 合并a和b所在的两个集合
69+
p[find(a)] = find(b)
70+
```
71+
72+
模板 2——维护 size 的并查集:
73+
74+
```python
75+
# 初始化,p存储每个点的父节点,size只有当节点是祖宗节点时才有意义,表示祖宗节点所在集合中,点的数量
76+
p = list(range(n))
77+
size = [1] * n
78+
79+
# 返回x的祖宗节点
80+
def find(x):
81+
if p[x] != x:
82+
# 路径压缩
83+
p[x] = find(p[x])
84+
return p[x]
85+
86+
# 合并a和b所在的两个集合
87+
if find(a) != find(b):
88+
size[find(b)] += size[find(a)]
89+
p[find(a)] = find(b)
90+
```
91+
92+
模板 3——维护到祖宗节点距离的并查集:
93+
94+
```python
95+
# 初始化,p存储每个点的父节点,d[x]存储x到p[x]的距离
96+
p = list(range(n))
97+
d = [0] * n
98+
99+
# 返回x的祖宗节点
100+
def find(x):
101+
if p[x] != x:
102+
t = find(p[x])
103+
d[x] += d[p[x]]
104+
p[x] = t
105+
return p[x]
106+
107+
# 合并a和b所在的两个集合
108+
p[find(a)] = find(b)
109+
d[find(a)] = distance
110+
```
111+
112+
对于本题,先遍历所有字符串对,判断两字符串是否相似,若相似,则将两字符串合并到同一个集合中,从而形成并查集。最后统计集合的数量即可。
113+
53114
<!-- tabs:start -->
54115

55116
### **Python3**
56117

57118
<!-- 这里可写当前语言的特殊实现逻辑 -->
58119

59120
```python
60-
121+
class Solution:
122+
def numSimilarGroups(self, strs: List[str]) -> int:
123+
n = len(strs)
124+
p = list(range(n))
125+
126+
def find(x):
127+
if p[x] != x:
128+
p[x] = find(p[x])
129+
return p[x]
130+
131+
def check(a, b):
132+
cnt = 0
133+
for i, c in enumerate(a):
134+
if c != b[i]:
135+
cnt += 1
136+
return cnt <= 2
137+
138+
for i in range(n):
139+
for j in range(i + 1, n):
140+
if check(strs[i], strs[j]):
141+
p[find(i)] = find(j)
142+
143+
return sum(i == find(i) for i in range(n))
61144
```
62145

63146
### **Java**
64147

65148
<!-- 这里可写当前语言的特殊实现逻辑 -->
66149

67150
```java
151+
class Solution {
152+
private int[] p;
153+
154+
public int numSimilarGroups(String[] strs) {
155+
int n = strs.length;
156+
p = new int[n];
157+
for (int i = 0; i < n; ++i) {
158+
p[i] = i;
159+
}
160+
for (int i = 0; i < n; ++i) {
161+
for (int j = i + 1; j < n; ++j) {
162+
if (check(strs[i], strs[j])) {
163+
p[find(i)] = find(j);
164+
}
165+
}
166+
}
167+
int res = 0;
168+
for (int i = 0; i < n; ++i) {
169+
if (i == find(i)) {
170+
++res;
171+
}
172+
}
173+
return res;
174+
}
175+
176+
private boolean check(String a, String b) {
177+
int cnt = 0;
178+
int n = a.length();
179+
for (int i = 0; i < n; ++i) {
180+
if (a.charAt(i) != b.charAt(i)) {
181+
++cnt;
182+
}
183+
}
184+
return cnt <= 2;
185+
}
186+
187+
private int find(int x) {
188+
if (p[x] != x) {
189+
p[x] = find(p[x]);
190+
}
191+
return p[x];
192+
}
193+
}
194+
```
195+
196+
### **C++**
197+
198+
```cpp
199+
class Solution {
200+
public:
201+
vector<int> p;
202+
int numSimilarGroups(vector<string>& strs) {
203+
int n = strs.size();
204+
for (int i = 0; i < n; ++i) p.push_back(i);
205+
for (int i = 0; i < n; ++i)
206+
{
207+
for (int j = i + 1; j < n; ++j)
208+
{
209+
if (check(strs[i], strs[j]))
210+
p[find(i)] = find(j);
211+
}
212+
}
213+
int res = 0;
214+
for (int i = 0; i < n; ++i)
215+
{
216+
if (i == find(i)) ++res;
217+
}
218+
return res;
219+
}
220+
221+
bool check(string a, string b) {
222+
int cnt = 0;
223+
int n = a.size();
224+
for (int i = 0; i < n; ++i)
225+
if (a[i] != b[i]) ++cnt;
226+
return cnt <= 2;
227+
}
228+
229+
int find(int x) {
230+
if (p[x] != x) p[x] = find(p[x]);
231+
return p[x];
232+
}
233+
};
234+
```
68235
236+
### **Go**
237+
238+
```go
239+
var p []int
240+
241+
func numSimilarGroups(strs []string) int {
242+
n := len(strs)
243+
p = make([]int, n)
244+
for i := 0; i < n; i++ {
245+
p[i] = i
246+
}
247+
for i := 0; i < n; i++ {
248+
for j := i + 1; j < n; j++ {
249+
if !check(strs[i], strs[j]) {
250+
continue
251+
}
252+
p[find(i)] = find(j)
253+
}
254+
}
255+
res := 0
256+
for i := 0; i < n; i++ {
257+
if i == find(i) {
258+
res++
259+
}
260+
}
261+
return res
262+
}
263+
264+
func check(a, b string) bool {
265+
cnt, n := 0, len(a)
266+
for i := 0; i < n; i++ {
267+
if a[i] != b[i] {
268+
cnt++
269+
}
270+
}
271+
return cnt <= 2
272+
}
273+
274+
func find(x int) int {
275+
if p[x] != x {
276+
p[x] = find(p[x])
277+
}
278+
return p[x]
279+
}
69280
```
70281

71282
### **...**
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
class Solution {
2+
public:
3+
vector<int> p;
4+
int numSimilarGroups(vector<string>& strs) {
5+
int n = strs.size();
6+
for (int i = 0; i < n; ++i) p.push_back(i);
7+
for (int i = 0; i < n; ++i)
8+
{
9+
for (int j = i + 1; j < n; ++j)
10+
{
11+
if (check(strs[i], strs[j]))
12+
p[find(i)] = find(j);
13+
}
14+
}
15+
int res = 0;
16+
for (int i = 0; i < n; ++i)
17+
{
18+
if (i == find(i)) ++res;
19+
}
20+
return res;
21+
}
22+
23+
bool check(string a, string b) {
24+
int cnt = 0;
25+
int n = a.size();
26+
for (int i = 0; i < n; ++i)
27+
if (a[i] != b[i]) ++cnt;
28+
return cnt <= 2;
29+
}
30+
31+
int find(int x) {
32+
if (p[x] != x) p[x] = find(p[x]);
33+
return p[x];
34+
}
35+
};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
var p []int
2+
3+
func numSimilarGroups(strs []string) int {
4+
n := len(strs)
5+
p = make([]int, n)
6+
for i := 0; i < n; i++ {
7+
p[i] = i
8+
}
9+
for i := 0; i < n; i++ {
10+
for j := i + 1; j < n; j++ {
11+
if !check(strs[i], strs[j]) {
12+
continue
13+
}
14+
p[find(i)] = find(j)
15+
}
16+
}
17+
res := 0
18+
for i := 0; i < n; i++ {
19+
if i == find(i) {
20+
res++
21+
}
22+
}
23+
return res
24+
}
25+
26+
func check(a, b string) bool {
27+
cnt, n := 0, len(a)
28+
for i := 0; i < n; i++ {
29+
if a[i] != b[i] {
30+
cnt++
31+
}
32+
}
33+
return cnt <= 2
34+
}
35+
36+
func find(x int) int {
37+
if p[x] != x {
38+
p[x] = find(p[x])
39+
}
40+
return p[x]
41+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,43 @@
1+
class Solution {
2+
private int[] p;
3+
4+
public int numSimilarGroups(String[] strs) {
5+
int n = strs.length;
6+
p = new int[n];
7+
for (int i = 0; i < n; ++i) {
8+
p[i] = i;
9+
}
10+
for (int i = 0; i < n; ++i) {
11+
for (int j = i + 1; j < n; ++j) {
12+
if (check(strs[i], strs[j])) {
13+
p[find(i)] = find(j);
14+
}
15+
}
16+
}
17+
int res = 0;
18+
for (int i = 0; i < n; ++i) {
19+
if (i == find(i)) {
20+
++res;
21+
}
22+
}
23+
return res;
24+
}
25+
26+
private boolean check(String a, String b) {
27+
int cnt = 0;
28+
int n = a.length();
29+
for (int i = 0; i < n; ++i) {
30+
if (a.charAt(i) != b.charAt(i)) {
31+
++cnt;
32+
}
33+
}
34+
return cnt <= 2;
35+
}
36+
37+
private int find(int x) {
38+
if (p[x] != x) {
39+
p[x] = find(p[x]);
40+
}
41+
return p[x];
42+
}
43+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,23 @@
1+
class Solution:
2+
def numSimilarGroups(self, strs: List[str]) -> int:
3+
n = len(strs)
4+
p = list(range(n))
5+
6+
def find(x):
7+
if p[x] != x:
8+
p[x] = find(p[x])
9+
return p[x]
10+
11+
def check(a, b):
12+
cnt = 0
13+
for i, c in enumerate(a):
14+
if c != b[i]:
15+
cnt += 1
16+
return cnt <= 2
17+
18+
for i in range(n):
19+
for j in range(i + 1, n):
20+
if check(strs[i], strs[j]):
21+
p[find(i)] = find(j)
22+
23+
return sum(i == find(i) for i in range(n))

0 commit comments

Comments
 (0)