翻转字符串中的单词

力扣题目链接(opens new window)

给定一个字符串,逐个翻转字符串中的每个单词。

示例 1:

输入: "the sky is blue"

输出: "blue is sky the"

示例 2:

输入: " hello world! "

输出: "world! hello"

解释: 输入字符串可以在前面或者后面包含多余的空格,但是反转后的字符不能包括。

示例 3:

输入: "a good example"

输出: "example good a"

解释: 如果两个单词间有多余的空格,将反转后单词间的空格减少到只含一个。

思路

以示例1为例,一种或很朴素的想法是:将字符串整个反转

"the sky is blue" ---> "eulb si yks eht"

然后再逐个将单词转回来

那么就有问题了,对于输入格式规范的字符串我们可以想上面那样处理

但是题目所给的输入有可能是不规范的

也就是说,字符串的前面、后间都有可能插入多个空格,反转之后还要将这些空格去除(也就是把格式统一为标准形式)

这就有点难搞

现在大致可以把解题思路分为三部分:

​ 1、将字符串转换为标准形式(去除多余的空格,仅保留单词间的必要空格)

​ 2、将标准字符串整体反转

​ 3、将反转字符串中的单词反转回正常顺序

代码

按照思路来看,都写在主函数里不太现实,可以先把“去除空格”函数和“反转字符串”函数分别写出来

去除空格函数

就根据思路来写

和以前一样,仍需要明确快慢指针的用途

void deleteExtraSpaces(string& s){
//定义慢指针
int slow = 0;
//遍历字符串
for(int fast = 0; fast < s.size(); ++fast){//++i和i++仅在效率上不同
if(s[fast] != ' '){
//如果当前字符(单词)不是第一个字符(单词),那么要在单词前面加空格
if(slow != 0){
s[slow] = ' ';
slow++;
}
//此时找到了单词的开头,需要将其按字符逐个移动到slow所指的位置
//循环条件中需要再判断当前fast是否指向空格,指到就停止移动
//进入下一个for循环,fast指针从道歉位置后移一位
while(fast < s.size() && s[fast] != ' '){
s[slow] = s[fast];
slow++;
fast++;
}
}
}
//因为去除了多余的空格,所以需要重新计算一下数组的大小
s.resize(slow);//此时slow即为标准化后字符串数组的长度
}

上述代码的过程图示如下:

这里 for 有个坑,后面注意事项的时候细说

反转函数

反转字符串 里面用的一样,当然也可以用c++提供的

注意:该函数可以反转一条字符串中指定位置的字符

void reverse(string& s, int start, int end){//注意要输入字符串的引用!!!
//在for中同时维护两个变量
for(int i = start, j = end; i < j; i++, j--){
swap(s[i], s[j]);
}
}
完整代码

在主函数里面使用这两个函数完成解题逻辑

class Solution {
public:
void deleteExtraSpaces(string& s){
//定义慢指针
int slow = 0;
//遍历字符串
for(int fast = 0; fast < s.size(); ++fast){//++i和i++仅在效率上不同
if(s[fast] != ' '){
//如果当前字符(单词)不是第一个字符(单词),那么要在单词前面加空格
if(slow != 0){
s[slow] = ' ';
slow++;
}
//此时找到了单词的开头,需要将其按字符逐个移动到slow所指的位置
//循环条件中需要再判断当前fast是否指向空格,指到就停止移动
//进入下一个for循环,fast指针从道歉位置后移一位
while(fast < s.size() && s[fast] != ' '){
s[slow] = s[fast];
slow++;
fast++;
}
}
}
//因为去除了多余的空格,所以需要重新计算一下数组的大小
s.resize(slow);//此时slow即为标准化后字符串数组的长度
} void reverse(string& s, int start, int end){
//在for中同时维护两个变量
for(int i = start, j = end; i < j; i++, j--){
swap(s[i], s[j]);
}
} string reverseWords(string s) {
//1、将字符串转换为标准形式,去除空格
deleteExtraSpaces(s);
//2、字符串整体反转
reverse(s, 0, s.size() - 1);//记得减1
//3、恢复每个单词的正常顺序
//依旧使用快慢指针
int slow = 0;//start,fast即为end
for(int fast = 0; fast <= s.size(); ++fast){
if(fast == s.size() || s[fast] == ' '){//到达空格或者字符串结尾倒要反转,不然会漏翻
reverse(s, slow, fast - 1);
slow = fast + 1;//从下一个单词的头开始
}
}
return s;
}
};

正确的思考方式应该是想清楚思路之后,实现每个部分需要的函数,然后再一块搭起来

注意点

1、for循环的特性

本题中大量且灵活的体现了for循环的一些平时容易忽略的特性

循环变量
for(int i = 0; i < xxx; i++){

}

上述是一个典型的for循环结构

要注意的是,开头的这句for(int i = 0; i < xxx; i++)

它只对循环变量 i 进行初始化(赋初值),以及在每次{ }内代码执行完毕后对循环变量执行循环条件(即i++

至于{ }内发生了什么,for循环本身是不管的。

也就是说,如果我在{ }内对循环变量 i进行了操作,也是允许的

例如:

for(int i = 0; i < xxx; i++){
i = 5;
}//那么下轮循环,i 就从6开始,而不是1
++ii++

如果你了解前++和后++的区别的话,那你应该知道后++的实现需要借助一个临时变量temp

实际上,在for的日常使用中,使用++i还是i++在结果上是没有区别的

但是如果是刷题的话,++i的执行效率会好一些,仅此而已

详见

2、TBD

【LeetCode字符串#03】图解翻转字符串中的单词,以及对于for使用的说明的更多相关文章

  1. LeetCode 151:给定一个字符串,逐个翻转字符串中的每个单词 Reverse Words in a String

    公众号:爱写bug(ID:icodebugs) 翻转字符串里的单词 Given an input string, reverse the string word by word. 示例 1: 输入: ...

  2. [LeetCode] Reverse String II 翻转字符串之二

    Given a string and an integer k, you need to reverse the first k characters for every 2k characters ...

  3. 151. Reverse Words in a String翻转一句话中的单词

    [抄题]: Given an input string, reverse the string word by word. Example: Input: "the sky is blue& ...

  4. [LeetCode] 344. Reverse String 翻转字符串

    Write a function that reverses a string. The input string is given as an array of characters char[]. ...

  5. C#版(击败100.00%的提交) - Leetcode 151. 翻转字符串里的单词 - 题解

    版权声明: 本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. C#版 - L ...

  6. leetcode python翻转字符串里的单词

    # Leetcode 151 翻转字符串里的单词### 题目描述给定一个字符串,逐个翻转字符串中的每个单词. **示例1:** 输入: "the sky is blue" 输出: ...

  7. LeetCode 151 翻转字符串里的单词

    题目: 给定一个字符串,逐个翻转字符串中的每个单词. 示例 1: 输入: "the sky is blue" 输出: "blue is sky the" 示例 ...

  8. LeetCode 翻转字符串里的单词

    给定一个字符串,逐个翻转字符串中的每个单词. 示例 1: 输入: "the sky is blue" 输出: "blue is sky the" 示例 2: 输 ...

  9. 力扣(LeetCode)翻转字符串里的单词 个人题解

    给定一个字符串,逐个翻转字符串中的每个单词. 示例 1: 输入: "the sky is blue" 输出: "blue is sky the" 示例 2: 输 ...

  10. Java实现 LeetCode 151 翻转字符串里的单词

    151. 翻转字符串里的单词 给定一个字符串,逐个翻转字符串中的每个单词. 示例 1: 输入: "the sky is blue" 输出: "blue is sky th ...

随机推荐

  1. TensorFlow深度学习!构建神经网络预测股票价格!⛵

    作者:韩信子@ShowMeAI 深度学习实战系列:https://www.showmeai.tech/tutorials/42 TensorFlow 实战系列:https://www.showmeai ...

  2. 手记系列之三 ----- 关于使用Nginx的一些使用方法和经验

    前言 本篇文章主要介绍的关于本人在使用Nginx的一些使用方法和经验~ Nginx介绍 介绍 Nginx("engine x")是一款是由俄罗斯的程序设计师Igor Sysoev所 ...

  3. ui自动化测试数据复原遇到的坑——1、hibernate输出完整sql

    公司老项目使用SSH+informix+weblogic+IE开发,我们要做ui自动化测试,其中的测试数据复原,我打算通过hibernate输出sql,然后把插入.更新的sql改为delete或upd ...

  4. 2022春每日一题:Day 16

    题目:不同子串个数 这题需要利用后缀数组求出的height的性质,我们发现对于每个后缀,他的height后的所有子串就是算在答案里,因此答案只需要求出n-height[i]-sa[i]+1的和就可以了 ...

  5. 链表实现-回文palindrome判断

    1.数字回文判断(逆转,分离未位,砍掉个位,保存原来) s = s * 10 + a%10 a = a/10 2.字符串判断回文 package main //思路: 开发一个栈来来存放链表的上半段f ...

  6. HDLBits答案——Verification: Reading Simulations

    1 Finding bugs in code 1.1 Bugs mux2 module top_module ( input sel, input [7:0] a, input [7:0] b, ou ...

  7. 两行CSS让页面提升了近7倍渲染性能!

    前言 对于前端人员来讲,最令人头疼的应该就是页面性能了,当用户在访问一个页面时,总是希望它能够快速呈现在眼前并且是可交互状态.如果页面加载过慢,你的用户很可能会因此离你而去.所以页面性能对于前端开发者 ...

  8. 【Spark】Day01-入门、模块组成、4种运行模式详解及配置、案例实操(spark分析过程)

    一.概述 1.概念 基于内存的大数据分析计算引擎 2.特点 快速.通用.可融合性 3.Spark内置模块[腾讯8000台spark集群] Spark运行在集群管理器(Cluster Manager)上 ...

  9. 【CDH数仓】Day02:业务数仓搭建、Kerberos安全认证+Sentry权限管理、集群性能测试及资源管理、邮件报警、数据备份、节点添加删除、CDH的卸载

    五.业务数仓搭建 1.业务数据生成 建库建表gmall 需求:生成日期2019年2月10日数据.订单1000个.用户200个.商品sku300个.删除原始数据. CALL init_data('201 ...

  10. 【RocketMQ】主从同步实现原理

    主从同步的实现逻辑主要在HAService中,在DefaultMessageStore的构造函数中,对HAService进行了实例化,并在start方法中,启动了HAService: public c ...