

cpp
for (int i = 0; i < s.size(); ++i) {
if (s[i] != ' ') { // 第1步:发现单词开始
if (slow != 0) s[slow++] = ' '; // 第2步:如果需要,先加空格(此时不会重复加)
// 第3步:复制整个单词
while (i < s.size() && s[i] != ' ') {
s[slow++] = s[i++]; // 注意:这里 i 也在增加!
}
}
// for循环的 ++i 会执行,但此时 i 可能已经超过边界或指向空格
}
初始状态: s = " h e l l o w o r l d"
0 1 2 3 4 5 6 7 8 9 10 11
i = 0, slow = 0, s[0] = 'h'(非空格)
执行步骤:
1. if (s[0] != ' ') 成立
2. if (slow != 0) → slow=0,所以不执行(不添加空格)
3. 进入 while 循环:
第1轮:i=0, s[0]='h' → s[0]=s[0], slow=1, i=1
第2轮:i=1, s[1]='e' → s[1]=s[1], slow=2, i=2
第3轮:i=2, s[2]='l' → s[2]=s[2], slow=3, i=3
第4轮:i=3, s[3]='l' → s[3]=s[3], slow=4, i=4
第5轮:i=4, s[4]='o' → s[4]=s[4], slow=5, i=5
第6轮:i=5, s[5]=' '(空格)→ while 条件不成立,退出循环
此时:i=5(指向空格),slow=5
for循环的 ++i 执行 → i=6
关键点:
-
while循环中的s[slow++] = s[i++]让i和slow同步前进 -
当遇到空格时,
while循环立即停止,此时i已经指向空格位置 -
然后
for循环的++i让i指向下一个字符
cpp
void resize(size_t n); // 基本形式
resize() 是 C++ 标准库中 std::string 类的一个成员函数,用于改变字符串的长度。
-
n:新的字符串长度(size_t类型,即无符号整数) -
如果不resize,原字符串后面可能还有多余的空格字符
-
后续的reverse操作会操作到这些无效字符,导致错误结果
cpp
//举例
string s = "hello world";
s.resize(5); // s 变成 "hello"
// 索引 5 及之后的字符被丢弃
cpp
class Solution {
public:
// 翻转字符串中指定区间的字符 [start, end] 闭区间
void reverse(string& s, int start, int end) {
// 使用双指针从两端向中间交换字符
for (int i = start, j = end; i < j; i++, j--) {
swap(s[i], s[j]); // 交换 s[i] 和 s[j] 位置的字符
}
}
// 去除字符串中多余的空格(移除前后及中间多余空格,只保留单词间一个空格)
void removeExtraSpaces(string& s) {
int slow = 0; // 慢指针,指向当前处理后字符串应存放的位置
for (int i = 0; i < s.size(); ++i) { // i 为快指针,遍历原字符串
if (s[i] != ' ') { // 遇到非空格字符(即单词部分)
if (slow != 0) s[slow++] = ' '; // 如果不是第一个单词,先加一个空格分隔
// 将当前单词完整拷贝到慢指针位置
while (i < s.size() && s[i] != ' ') {
s[slow++] = s[i++];
}
}
// 遇到空格则跳过,由上面的 if 条件处理
}
s.resize(slow); // 调整字符串大小,去除多余部分
}
// 主函数:反转字符串中的单词顺序
string reverseWords(string s) {
// 第一步:去除多余空格,规范单词间隔
removeExtraSpaces(s);
// 第二步:翻转整个字符串
reverse(s, 0, s.size() - 1);
// 第三步:逐个单词翻转回来,恢复单词内部顺序
int start = 0; // 记录当前单词的起始位置
for (int i = 0; i <= s.size(); ++i) {
// 遇到空格或字符串结尾,说明一个单词结束
if (i == s.size() || s[i] == ' ') {
reverse(s, start, i - 1); // 翻转当前单词
start = i + 1; // 更新下一个单词起始位置(跳过空格)
}
}
return s; // 返回处理后的字符串
}
};