当前位置: 首页 > news >正文

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

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

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

方法思路

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

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

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

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

C++ 代码实现

#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. 边界处理

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

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

方法二:

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;}
};
http://www.lryc.cn/news/582354.html

相关文章:

  • TCP长连接保持在线状态
  • 人工智能-基础篇-23-智能体Agent到底是什么?怎么理解?(智能体=看+想+做)
  • 数据中台架构解析:湖仓一体的实战设计
  • 计算阶梯电费
  • C盘瘦身 -- 虚拟内存文件 pagefile.sys
  • Go defer(二):从汇编的角度理解延迟调用的实现
  • MIL-STD-1553B总线
  • NLP自然语言处理 02 RNN及其变体
  • 【Java面试】Https和Http的区别?以及分别的原理是什么?
  • 【应急响应】Linux 自用应急响应工具(LinuxCheckShoot)
  • 【力扣(LeetCode)】数据挖掘面试题0003: 356. 直线镜像
  • 明星AI自动化测试工具Midscene.js源码解析
  • Vidwall: 支持将 4K 视频设置为动态桌面壁纸,兼容 MP4 和 MOV 格式
  • 【保姆级图文详解】探秘 Prompt 工程:AI 交互的关键密码
  • 【Netty基础】Java原生网络编程
  • 熔断限流降级
  • [附源码+数据库+毕业论文]基于Spring+MyBatis+MySQL+Maven+jsp实现的高校实验室资源综合管理系统,推荐!
  • Spring @Conditional注解深度解析:从原理到最佳实践
  • 10.6 ChatGLM3私有数据微调实战:24小时打造高精度模型,显存直降60%
  • 【机器学习笔记 Ⅲ】4 特征选择
  • 【ARM AMBA AXI 入门 21 -- AXI partial 访问和 narrow 访问的区别】
  • 田间杂草分割实例
  • Qt的第一个程序(2)
  • JVM基础01(从入门到八股-黑马篇)
  • 微信小程序81~90
  • C++笔记之和的区别
  • 力扣 hot100 Day37
  • 回溯题解——子集【LeetCode】二进制枚举法
  • ubuntu18.04.1无法安装vscode(安装依赖无效)
  • qiankun 微前端框架子应用间通信方法详解