LeetCode 151. 反转字符串中的单词

LeetCode 151. 反转字符串中的单词 解析

这个问题要求将给定字符串中的单词顺序反转,同时移除多余空格,使得每个单词之间只有一个空格,且字符串首尾无空格。

方法思路

最优解法是原地反转法:通过三次反转操作实现单词顺序的反转,同时处理多余空格。具体步骤如下:

  1. 移除多余空格

    将字符串中的多余空格去除,只保留单词间的单个空格,并调整字符串长度。

  2. 整体反转

    将处理后的字符串整体反转。

  3. 单词反转

    遍历字符串,对每个单词进行局部反转,恢复单词的原始顺序。

C++ 代码实现

cpp 复制代码
#include <string>
#include <algorithm>

class Solution {
public:
    string reverseWords(string s) {
        // 1. 移除多余空格
        int i = 0, j = 0;  // i 为写入指针,j 为读取指针
        while (j < s.length()) {
            // 跳过前导空格
            while (j < s.length() && s[j] == ' ') j++;
            if (j >= s.length()) break;  // 处理全空格的情况
            
            // 添加单词和中间空格
            if (i != 0) s[i++] = ' ';  // 单词间添加一个空格
            while (j < s.length() && s[j] != ' ') {
                s[i++] = s[j++];
            }
        }
        s.resize(i);  // 调整字符串长度
        
        // 2. 整体反转
        reverse(s.begin(), s.end());
        
        // 3. 逐个单词反转
        i = 0;
        for (int j = 0; j <= s.length(); j++) {
            if (j == s.length() || s[j] == ' ') {  // 遇到空格或字符串末尾
                reverse(s.begin() + i, s.begin() + j);
                i = j + 1;  // 更新下一个单词的起始位置
            }
        }
        
        return s;
    }
};

代码解释

  1. 移除多余空格

    • 使用双指针 ij,其中 j 用于遍历原字符串,i 用于写入处理后的字符。
    • 跳过前导空格,遇到单词时将其复制到 i 位置,并在单词间添加单个空格。
    • 最终使用 resize(i) 调整字符串长度,去除尾部冗余字符。
  2. 整体反转

    使用 reverse(s.begin(), s.end()) 将整个字符串反转,此时单词顺序已反转,但每个单词内部字符顺序也被反转。

  3. 单词反转

    • 遍历字符串,每当遇到空格或字符串末尾时,将当前单词(从 ij-1)进行局部反转。
    • 更新 i 为下一个单词的起始位置(j+1)。

复杂度分析

  • 时间复杂度:O(n),其中 n 是字符串的长度。每个字符最多被处理三次(移除空格、整体反转、单词反转)。
  • 空间复杂度:O(1),仅需常数级的额外空间。

示例

输入:s = " hello world! "

输出:"world! hello"
解释

  1. 移除多余空格"hello world!"
  2. 整体反转"!dlrow olleh"
  3. 单词反转"world! hello"

关键点

  1. 原地修改

    通过双指针法在原字符串上直接操作,避免额外空间开销。

  2. 三次反转策略

    整体反转+单词反转的组合巧妙地实现了单词顺序的反转,同时保持单词内容正确。

  3. 边界处理

    • 处理字符串首尾的多余空格。
    • 处理连续多个空格的情况。
    • 处理空字符串或全空格字符串的情况。

这种方法高效且符合题目要求,是解决此类问题的经典思路。

方法二:

cpp 复制代码
class Solution {
public:
    string reverseWords(string s) {
        stringstream ss(s);

        vector<string> words;
        string word;

        while (ss >> word)
        {
            words.emplace_back(word);
        }
        
        string res = words[words.size() - 1];
        for (int i = words.size() - 2; i >= 0; --i)
        {
            res += " ";
            res += words[i];
        }

        return res;

    }
};
相关推荐
Vic1010122 分钟前
Hutool 的完整 JSON 工具类示例
开发语言·json
程序员编程指南27 分钟前
Qt 开发 IDE 插件开发指南
c语言·c++·ide·qt·elasticsearch
蹦蹦跳跳真可爱58942 分钟前
Python----MCP(MCP 简介、uv工具、创建MCP流程、MCP客户端接入Qwen、MCP客户端接入vLLM)
开发语言·人工智能·python·语言模型
SKYDROID云卓小助手43 分钟前
无人设备遥控器之多设备协同技术篇
网络·人工智能·嵌入式硬件·算法·信号处理
MediaTea1 小时前
Python 库手册:getopt Unix 风格参数解析模块
服务器·开发语言·python·unix
王尼莫啊1 小时前
【立体标定】圆形标定板标定python实现
开发语言·python·opencv
熬了夜的程序员1 小时前
【华为机试】34. 在排序数组中查找元素的第一个和最后一个位置
数据结构·算法·华为od·华为·面试·golang
laocooon5238578861 小时前
C语言画表格
c语言·开发语言
phltxy1 小时前
ArrayList与顺序表
java·算法
Entropy-Lee2 小时前
JavaScript 语句和函数
开发语言·前端·javascript