zoukankan      html  css  js  c++  java
  • 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 的代码

  • 相关阅读:
    UVaLive 7362 Farey (数学,欧拉函数)
    UVaLive 7361 Immortal Porpoises (矩阵快速幂)
    UVaLive 7359 Sum Kind Of Problem (数学,水题)
    CodeForces 706D Vasiliy's Multiset (字典树查询+贪心)
    负载均衡服务器
    集群-如何理解集群?
    架构规划
    领域模型
    状态图
    E-R图
  • 原文地址:https://www.cnblogs.com/abc-begin/p/8231066.html
Copyright © 2011-2022 走看看