输出:
happwnewwear
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
用例四:
输入:
()abcdefgAC(a)(Ab)(C)
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
输出:
AAcdefgAC
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
说明 等效字符集为(‘a’, ‘A’, ‘b’),输入字符里没有被小括号包含的子字符串集合为"abcdefgAC",将其中字符替换为字典序最小的等效字符后输出为:“AAcdefgAC”
python解法
输入一个包含普通字符和括号的字符串,括号内的字符代表一组等价字符。例如,输入 (ab)c(d)e 表示:
‘a’ 和 ‘b’ 等价;
‘d’ 等价于自身;
外部字符 ‘c’ 和 ‘e’ 与括号中的字符没有关系。
任务是将字符串中所有等价字符替换为其字典序最小的字符,并输出替换后的字符串。
如果输入中没有普通字符,则返回 0。
实现步骤:
解析输入:
遍历字符串,将括号中的字符分组存入等价集合 equivalent_sets。
将不在括号内的字符存入 body_chars。
处理等价集合:
合并等价集合:如果两个集合中有字符相同或字符大小写等价(如 ‘a’ 和 ‘A’),则将它们合并为一个集合。
字符替换:
遍历等价集合,将集合中的字符替换为集合中字典序最小的字符。
遍历 body_chars,根据等价集合中的替换规则更新其值。
输出结果:
将处理后的字符拼接为字符串并返回。
如果没有普通字符,返回 0。
核心逻辑:
合并等价集合:两个集合是否合并取决于它们是否包含相同的字符或大小写等价字符。
替换规则:对于每个等价集合,用集合中字典序最小的字符替换所有等价字符。
时间复杂度:
解析输入:O(n),其中 n 为字符串长度。
合并集合:最坏情况下 O(m² * k),其中 m 是等价集合的数量,k 是单个集合的最大长度。
替换字符:O(n)。
class CharacterProcessor:
def __init__(self):
self.equivalent_sets = []
self.body_chars = []
def process_input(self, s):
"""
解析输入字符串,将括号中的字符分组为等价集合,将普通字符存入 body_chars。
"""
is_open = False
for c in s:
if c == '(':
is_open = True
self.equivalent_sets.append(set())
elif c == ')':
is_open = False
if len(self.equivalent_sets[-1]) == 0:
self.equivalent_sets.pop()
else:
if not is_open:
self.body_chars.append(c)
else:
self.equivalent_sets[-1].add(c)
def can_combine(self, set1, set2):
"""
检查两个集合是否可以合并(有共同字符或字符大小写等价)。
"""
for c in range(97, 123):
lc = chr(c)
uc = chr(c - 32)
if (lc in set1 or uc in set1) and (lc in set2 or uc in set2):
return True
return False
def merge_equivalents(self):
"""
合并等价集合,直到无法再合并。
"""
while self._perform_merge():
pass
def _perform_merge(self):
"""
执行一次等价集合的合并操作。
"""
for i in range(len(self.equivalent_sets)):
for j in range(i + 1, len(self.equivalent_sets)):
if self.can_combine(self.equivalent_sets[i], self.equivalent_sets[j]):
tmp = list(self.equivalent_sets[i])
tmp.extend(self.equivalent_sets[j])
self.equivalent_sets[i] = set(tmp)
self.equivalent_sets.pop(j)
return True
return False
def replace_with_min_characters(self):
"""
使用等价集合中的最小字符替换所有等价字符。
"""
for eq in self.equivalent_sets:
tmp = list(eq)
tmp.sort()
min_char = tmp[0]
for i in range(len(self.body_chars)):
if self.body_chars[i] in eq:
self.body_chars[i] = min_char
def get_result(self):
"""
返回处理后的字符串结果。如果没有普通字符,返回 "0"。
"""
return "".join(self.body_chars) if self.body_chars else "0"
def getResult(s):
"""
主函数:解析字符串,处理等价关系并返回结果。
"""
processor = CharacterProcessor()
processor.process_input(s)
processor.merge_equivalents()
processor.replace_with_min_characters()
return processor.get_result()
if __name__ == "__main__":
input_string = input()
print(getResult(input_string))
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 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
java解法
输入一个字符串,包含括号中的字符等价关系。
括号中的字符表示等价,多个括号的关系通过联合并查集(Union-Find)进行合并。
外部的字符按原顺序保留,最终将所有字符替换为它们等价类中的字典序最小字符。
如果没有外部字符,返回 “0”。
解决方法:
Union-Find 解决等价关系:
使用并查集(Union-Find)结构存储字符的等价关系。
每个字符初始化为一个独立的集合,表示自身等价。
遇到括号内的字符时,合并这些字符的集合。
字符映射:
使用一个大小为 52 的并查集(a-z 和 A-Z 分别映射到 0-25 和 26-51)。
字符通过索引映射到并查集中的位置。
结果替换:
遍历字符串,将每个字符替换为它等价类中的字典序最小字符。
输入输出:
输入:
一个字符串,包含普通字符和括号中的等价关系。
输出:
字符串中所有字符替换为等价类中最小字符后的结果;
如果没有普通字符,返回 “0”。
核心逻辑:
Union-Find:
每个字符初始化为独立集合。
使用 union 合并等价字符集合。
使用 find 查找字符所属集合,并获取集合中字典序最小的字符。
字符串处理:
遍历字符串:
括号内的字符加入等价集合。
普通字符直接加入结果。
替换普通字符为等价类中的最小字符
import java.util.*;
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
String input = scanner.nextLine();
System.out.println(simplifyString(input));
}
public static String simplifyString(String s) {
UnionFind uf = new UnionFind(52);
StringBuilder result = new StringBuilder();
boolean insideBracket = false;
Set<Character> seenChars = new HashSet<>();
for (int i = 0; i < s.length(); i++) {
char c = s.charAt(i);
if (c == '(') {
insideBracket = true;
} else if (c == ')') {
insideBracket = false;
seenChars.clear();
} else if (insideBracket) {
seenChars.add(c);
for (char seen : seenChars) {
uf.union(seen, c);
}
} else {
result.append(c);
}
}
char[] finalResult = result.toString().toCharArray();
for (int i = 0; i < finalResult.length; i++) {
finalResult[i] = uf.findMinEquivalent(finalResult[i]);
}
String simplified = new String(finalResult);
return simplified.isEmpty() ? "0" : simplified;
}
}
class UnionFind {
private int[] parent;
private char[] minEquivalent;
public UnionFind(int size) {
parent = new int[size];
minEquivalent = new char[size];
for (int i = 0; i < size; i++) {
parent[i] = i;
minEquivalent[i] = (char) (i < 26 ? 'a' + i : 'A' + (i - 26));
}
}
public int find(int x) {
if (parent[x] != x) {
parent[x] = find(parent[x]);
minEquivalent[x] = minEquivalent[parent[x]];
}
return parent[x];
}
public void union(char a, char b) {
int rootA = find(charToIndex(a));
int rootB = find(charToIndex(b));
if (rootA != rootB) {
if (minEquivalent[rootA] < minEquivalent[rootB]) {
parent[rootB] = rootA;
minEquivalent[rootA] = (char) Math.min(minEquivalent[rootA], minEquivalent[rootB]);
} else {
parent[rootA] = rootB;
minEquivalent[rootB] = (char) Math.min(minEquivalent[rootA], minEquivalent[rootB]);
}
}
}
public char findMinEquivalent(char c) {
return minEquivalent[find(charToIndex(c))];
}
private int charToIndex(char c) {
return c >= 'a' ? c - 'a' : c - 'A' + 26;
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 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
C++解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
C解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
JS解法
输入字符串包含普通字符和括号中的字符等价关系。
括号中的字符等价(如 (ab) 表示 a 和 b 是等价的)。
字符串中的每个字符需替换为其等价类中字典序最小的字符。
如果输入中没有普通字符,输出 “0”。
解决方案:
使用并查集(Union-Find)存储字符的等价关系。
每个字符初始化为独立集合,通过 union 操作合并等价字符。
遍历普通字符并通过并查集找到它们的等价类中的字典序最小字符。
拼接替换后的字符并返回结果。
实现步骤:
初始化并查集:
使用对象 parent 存储字符的父节点。
find 方法:找到字符的根节点,并路径压缩。
union 方法:将两个字符的根节点合并为一个。
解析字符串:
遍历字符串:
遇到 ( 开始记录括号内字符。
遇到 ) 合并括号内字符的等价关系。
普通字符直接加入结果数组。
替换字符:
遍历结果数组,将每个字符替换为并查集中的字典序最小字符。
输出结果:
如果结果数组为空,返回 “0”。
时间复杂度:
解析字符串:O(n),其中 n 是字符串长度。
并查集操作:近似 O(1)(由于路径压缩和按秩合并)。
字符替换:O(n)
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
rl.on("line", (line) => {
console.log(simplifyString(line));
});
function simplifyString(s) {
const parent = {};
function find(x) {
if (parent[x] === undefined) {
parent[x] = x;
}
if (parent[x] !== x) {
parent[x] = find(parent[x]);
}
return parent[x];
}
function union(x, y) {
const rootX = find(x);
const rootY = find(y);
if (rootX !== rootY) {
if (rootX < rootY) parent[rootY] = rootX;
else parent[rootX] = rootY;
}
}
const cArr = [];
let isOpen = false;
let eqSet = [];
for (let i = 0; i < s.length; i++) {
const c = s[i];
if (c === '(') {
isOpen = true;
eqSet = [];
} else if (c === ')') {
isOpen = false;
for (let ch1 of eqSet) {
for (let ch2 of eqSet) {
union(ch1.toLowerCase(), ch2.toLowerCase());
}
}
} else {
if (!isOpen) {
cArr.push(c);
} else {
eqSet.push(c);
}
}
}
const ans = cArr.map(c => find(c.toLowerCase())).join("");
return ans.length === 0 ? "0" : ans;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 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
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: