abc_begin

导航

53. Reverse Words in a String【easy】

Given an input string, reverse the string word by word.

For example,
Given s = "the sky is blue",
return "blue is sky the".

 
Clarification
  • What constitutes a word?
    A sequence of non-space characters constitutes a word.
  • Could the input string contain leading or trailing spaces?
    Yes. However, your reversed string should not contain leading or trailing spaces.
  • How about multiple spaces between two words?
    Reduce them to a single space in the reversed string.

 

题意

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

 

说明

  • 单词的构成:无空格字母构成一个单词
  • 输入字符串是否包括前导或者尾随空格?可以包括,但是反转后的字符不能包括
  • 如何处理两个单词间的多个空格?在反转字符串中间空格减少到只含一个

 

解法一:

 1 class Solution {
 2     /**
 3      * @param s : A string
 4      * @return : A string
 5      */
 6 public:
 7     string reverseWords(string s) {
 8         string ss;
 9         int i = s.length() - 1;
10         while (i >= 0) {
11             while (i >= 0 && s[i] == ' ') {
12                 i--;
13             }
14             
15             if (i < 0) {
16                 break;  
17             }
18             
19             if (ss.length() != 0) {
20                 ss.push_back(' '); 
21             }
22                 
23             string temp ;
24             for (; i >= 0 && s[i] != ' '; i--) {
25                 temp.push_back(s[i]);    
26             }
27                 
28             reverse(temp.begin(),temp.end());
29             
30             ss.append(temp);
31         }
32        
33         return ss;
34     }
35 };

1、从后向前行进,如果为空格,则下标就递减,这是为了过滤最后面的那些空格。

2、遇到非空格就放到temp中,然后再翻转放入ss中

3、下一次再进入循环,还是过滤多余的空格,然后如果不是第一次进入循环就需要补充一个空格进来,再重复上面的过程即可

 

解法二:

 1 class Solution {
 2     /**
 3      * @param s : A string
 4      * @return : A string
 5      */
 6 public:
 7     string reverseWords(string s) {
 8         int storeIndex = 0, n = s.size();
 9         
10         reverse(s.begin(), s.end());
11         
12         for (int i = 0; i < n; ++i) {
13             if (s[i] != ' ') {
14                 if (storeIndex != 0) {
15                     s[storeIndex++] = ' ';
16                 }
17                 
18                 int j = i;
19                 while (j < n && s[j] != ' ') {
20                     s[storeIndex++] = s[j++];
21                 }
22                 
23                 reverse(s.begin() + storeIndex - (j - i), s.begin() + storeIndex);
24                 
25                 i = j;
26             }
27         }
28         s.resize(storeIndex);
29        
30         return s;
31     }
32 };

先整个字符串整体翻转一次,然后再分别翻转每一个单词(或者先分别翻转每一个单词,然后再整个字符串整体翻转一次),此时就能得到我们需要的结果了。那么这里我们需要定义一些变量来辅助我们解题,storeIndex表示当前存储到的位置,n为字符串的长度。我们先给整个字符串反转一下,然后我们开始循环,遇到空格直接跳过,如果是非空格字符,我们此时看storeIndex是否为0,为0的话表示第一个单词,不用增加空格;如果不为0,说明不是第一个单词,需要在单词中间加一个空格,然后我们要找到下一个单词的结束位置我们用一个while循环来找下一个为空格的位置,在此过程中继续覆盖原字符串,找到结束位置了,下面就来翻转这个单词,然后更新i为结尾位置,最后遍历结束,我们剪裁原字符串到storeIndex位置,就可以得到我们需要的结果。

参考@grandyang 的代码

 

解法三:

 1 class Solution {
 2     /**
 3      * @param s : A string
 4      * @return : A string
 5      */
 6 public:
 7     string reverseWords(string s) {
 8         istringstream is(s);
 9         string tmp;
10         is >> s;
11         while (is >> tmp) {
12             s = tmp + " " + s;
13         }
14         if (!s.empty() && s[0] == ' ') {
15             s = "";
16         }
17        
18         return s;
19     }
20 };

先把字符串装载入字符串流中,然后定义一个临时变量tmp,然后把第一个单词赋给s,这里需要注意的是,如果含有非空格字符,那么每次>>操作就会提取连在一起的非空格字符,那么我们每次将其加在s前面即可;如果原字符串为空,那么就不会进入while循环;如果原字符串为许多空格字符连在一起,那么第一个>>操作就会提取出这些空格字符放入s中,然后不进入while循环,这时候我们只要判断一下s的首字符是否为空格字符,是的话就将s清空即可。

参考@Zhoujingjin 的代码

 

解法四:

 1 class Solution {
 2     /**
 3      * @param s : A string
 4      * @return : A string
 5      */
 6 public:
 7     string reverseWords(string s) {
 8         istringstream is(s);
 9         s = "";
10         string t = "";
11         
12         while (getline(is, t, ' ')) {
13             if (t.empty()) {
14                 continue;
15             }
16             s = (s.empty() ? t : (t + " " + s));
17         }
18         
19         return s;
20     }
21 };

使用getline来做,第三个参数是设定分隔字符,我们用空格字符来分隔,这个跟上面的>>操作是有不同的,每次只能过一个空格字符,如果有多个空格字符连在一起,那么t会赋值为空字符串,所以我们在处理t的时候首先要判断其是否为空,是的话直接跳过。

参考@grandyang 的代码

 

posted on 2018-01-07 19:34  LastBattle  阅读(267)  评论(0编辑  收藏  举报