leetcode算法(151.反转字符串中的单词)

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++]islow 同步前进

  • 当遇到空格时,while循环立即停止,此时 i 已经指向空格位置

  • 然后 for 循环的 ++ii 指向下一个字符

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;  // 返回处理后的字符串
    }
};
相关推荐
ada7_18 小时前
LeetCode(python)78.子集
开发语言·数据结构·python·算法·leetcode·职场和发展
DeepVis Research19 小时前
【AGI/Simulation】2026年度通用人工智能图灵测试与高频博弈仿真基准索引 (Benchmark Index)
大数据·人工智能·算法·数据集·量化交易
努力学算法的蒟蒻19 小时前
day52(1.3)——leetcode面试经典150
算法·leetcode·面试
leoufung19 小时前
LeetCode 97. 交错字符串 - 二维DP经典题解(C语言实现)
c语言·算法·leetcode
leiming621 小时前
c++ map容器
开发语言·c++·算法
杨校1 天前
杨校老师课堂备赛C++信奥之模拟算法习题专项训练
开发语言·c++·算法
世洋Blog1 天前
AStar算法基础学习总结
算法·面试·c#·astar·寻路
haing20191 天前
七轴协作机器人运动学正解计算方法
算法·机器学习·机器人