LeetCode 151-反转字符串中的单词
题目描述
给你一个字符串 s ,请你反转字符串中 单词 的顺序。
单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开。
返回 单词 顺序颠倒且 单词 之间用单个空格连接的结果字符串。
注意:输入字符串 s中可能会存在前导空格、尾随空格或者单词间的多个空格。返回的结果字符串中,单词间应当仅用单个空格分隔,且不包含任何额外的空格。
解题思路
判断:
- 我的思路:使用辅助空间,用split库函数,分隔单词,然后定义一个新的string字符串,最后再把单词倒序相加。(但这样无意义)
- 不使用辅助空间的解法,空间复杂度要求为 O(1)
- 移除多余空格
- 将整个字符串反转
- 将每个单词反转(在这里可以利用到上一个函数)
代码
class Solution {
public String reverseWords(String s) {
// 1.去除首尾以及中间多余空格
StringBuilder sb = removeSpace(s);
// 2.反转整个字符串
reverseString(sb, 0, sb.length() - 1);
// 3.反转各个单词
reverseEachWord(sb);
return sb.toString();
}
private StringBuilder removeSpace(String s){
int start = 0;
int end = s.length() - 1;
while (s.charAt(start) == ' ' ) start++;
while (s.charAt(end) == ' ' ) end--;
StringBuilder sb = new StringBuilder();
while (start <= end) {
char c = s.charAt(start);
if (c != ' ' || sb.charAt(sb.length() - 1) != ' ') {
sb.append(c);
}
start++;
}
return sb;
}
public void reverseString(StringBuilder sb, int start, int end) {
while (start < end) {
char temp = sb.charAt(start);
sb.setCharAt(start, sb.charAt(end));
sb.setCharAt(end, temp);
start++;
end--;
}
}
private void reverseEachWord(StringBuilder sb) {
int start = 0; // 每个单词的start
int end = 1; // 每个单词的end
int n = sb.length();
while (start < n) {
while (end < n && sb.charAt(end) != ' ') {
end++;
}
reverseString(sb, start, end - 1);
start = end + 1; // 因为此时end指向空格
end = start + 1;
}
}
}
还有其他三种解法,之后再看
复杂度
- 时间复杂度
O(n) - 空间复杂度
O(1) 或 O(n),取决于语言中字符串是否可变
难点
StringBuilder
类的用法- 使用
String
进行字符串拼接时,每次拼接都会创建一个新的字符串对象,将产生大量的临时对象,会影响性能。 StringBuilder
是可变的,允许我们在不创建新字符串对象的情况下进行字符串的连接和修改,不会创建大量的临时对象,因此更高效。
- 使用
总结
- 复习了
StringBuilder
类的使用 - 在写代码时,要多注意是否可以提高代码的复用率。