Given an input string, reverse the string word by word.
For example,
Given s = "the sky is blue
",
return "blue is sky the
".
- 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 的代码