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;  // 返回处理后的字符串
    }
};
相关推荐
AI软著研究员2 小时前
程序员必看:软著不是“面子工程”,是代码的“法律保险”
算法
FunnySaltyFish2 小时前
什么?Compose 把 GapBuffer 换成了 LinkBuffer?
算法·kotlin·android jetpack
颜酱3 小时前
理解二叉树最近公共祖先(LCA):从基础到变种解析
javascript·后端·算法
地平线开发者19 小时前
SparseDrive 模型导出与性能优化实战
算法·自动驾驶
董董灿是个攻城狮19 小时前
大模型连载2:初步认识 tokenizer 的过程
算法
地平线开发者20 小时前
地平线 VP 接口工程实践(一):hbVPRoiResize 接口功能、使用约束与典型问题总结
算法·自动驾驶
罗西的思考20 小时前
AI Agent框架探秘:拆解 OpenHands(10)--- Runtime
人工智能·算法·机器学习
HXhlx1 天前
CART决策树基本原理
算法·机器学习
Wect1 天前
LeetCode 210. 课程表 II 题解:Kahn算法+DFS 双解法精讲
前端·算法·typescript
颜酱1 天前
单调队列:滑动窗口极值问题的最优解(通用模板版)
javascript·后端·算法