欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 汽车 > 新车 > 《算法篇:翻转字符串里的单词 - Java 多种解法详解》

《算法篇:翻转字符串里的单词 - Java 多种解法详解》

2025/3/13 8:55:18 来源:https://blog.csdn.net/qq_67028830/article/details/146211601  浏览:    关键词:《算法篇:翻转字符串里的单词 - Java 多种解法详解》

题目描述

给定一个字符串,要求逐个翻转其中的每个单词。输入字符串可能在前后包含多余的空格,并且单词之间可能存在多个空格,但反转后的字符串不能包含多余空格,单词间应仅保留一个空格。

示例

  • 输入: "the sky is blue",输出: "blue is sky the"
  • 输入: " hello world! ",输出: "world! hello"
  • 输入: "a good example",输出: "example good a"

解法一:不使用 Java 内置方法实现

class Solution {/*** 不使用Java内置方法实现* <p>* 1.去除首尾以及中间多余空格* 2.反转整个字符串* 3.反转各个单词*/public String reverseWords(String s) {// 1.去除首尾以及中间多余空格StringBuilder sb = removeSpace(s);// 2.反转整个字符串reverseString(sb, 0, sb.length() - 1);// 3.反转各个单词reverseEachWord(sb);return sb.toString();}private StringBuilder removeSpace(String s) {int start = 0;int end = s.length() - 1;while (s.charAt(start) == ' ') start++;while (s.charAt(end) == ' ') end--;StringBuilder sb = new StringBuilder();while (start <= end) {char c = s.charAt(start);if (c != ' ' || sb.charAt(sb.length() - 1) != ' ') {sb.append(c);}start++;}return sb;}/*** 反转字符串指定区间[start, end]的字符*/public void reverseString(StringBuilder sb, int start, int end) {while (start < end) {char temp = sb.charAt(start);sb.setCharAt(start, sb.charAt(end));sb.setCharAt(end, temp);start++;end--;}}private void reverseEachWord(StringBuilder sb) {int start = 0;int end = 1;int n = sb.length();while (start < n) {while (end < n && sb.charAt(end) != ' ') {end++;}reverseString(sb, start, end - 1);start = end + 1;end = start + 1;}}
}
复杂度分析
  • 时间复杂度:(O(n)),其中 n 是字符串的长度。去除空格、反转字符串和反转单词的操作都需要遍历字符串一次。
  • 空间复杂:(O(n)),主要用于存储 StringBuilder 中的字符。

解法二:创建新字符数组填充

class Solution {public String reverseWords(String s) {//源字符数组char[] initialArr = s.toCharArray();//新字符数组char[] newArr = new char[initialArr.length + 1];//下面循环添加"单词 ",最终末尾的空格不会返回int newArrPos = 0;//i来进行整体对源字符数组从后往前遍历int i = initialArr.length - 1;while (i >= 0) {while (i >= 0 && initialArr[i] == ' ') {i--;}  //跳过空格//此时i位置是边界或!=空格,先记录当前索引,之后的while用来确定单词的首字母的位置int right = i;while (i >= 0 && initialArr[i] != ' ') {i--;}//指定区间单词取出(由于i为首字母的前一位,所以这里+1,),取出的每组末尾都带有一个空格for (int j = i + 1; j <= right; j++) {newArr[newArrPos++] = initialArr[j];if (j == right) {newArr[newArrPos++] = ' '; //空格}}}//若是原始字符串没有单词,直接返回空字符串;若是有单词,返回0-末尾空格索引前范围的字符数组(转成String返回)if (newArrPos == 0) {return "";} else {return new String(newArr, 0, newArrPos - 1);}}
}
复杂度分析
  • 时间复杂度:(O(n)),需要遍历源字符数组一次。
  • 空间复杂度:(O(n)),主要用于存储新字符数组。

解法三:双反转 + 移位

class Solution {/*** 思路:* ①反转字符串  "the sky is blue " => " eulb si yks eht"* ②遍历 " eulb si yks eht",每次先对某个单词进行反转再移位* 这里以第一个单词进行为演示:" eulb si yks eht" ==反转=> " blue si yks eht" ==移位=> "blue si yks eht"*/public String reverseWords(String s) {//步骤1:字符串整体反转(此时其中的单词也都反转了)char[] initialArr = s.toCharArray();reverse(initialArr, 0, s.length() - 1);int k = 0;for (int i = 0; i < initialArr.length; i++) {if (initialArr[i] == ' ') {continue;}int tempCur = i;while (i < initialArr.length && initialArr[i] != ' ') {i++;}for (int j = tempCur; j < i; j++) {if (j == tempCur) { //步骤二:二次反转reverse(initialArr, tempCur, i - 1); //对指定范围字符串进行反转,不反转从后往前遍历一个个填充有问题}//步骤三:移动操作initialArr[k++] = initialArr[j];if (j == i - 1) { //遍历结束//避免越界情况,例如=> "asdasd df f",不加判断最后就会数组越界if (k < initialArr.length) {initialArr[k++] = ' ';}}}}if (k == 0) {return "";} else {//参数三:以防出现如"asdasd df f"=>"f df asdasd"正好凑满不需要省略空格情况return new String(initialArr, 0, (k == initialArr.length) && (initialArr[k - 1] != ' ') ? k : k - 1);}}public void reverse(char[] chars, int begin, int end) {for (int i = begin, j = end; i < j; i++, j--) {chars[i] ^= chars[j];chars[j] ^= chars[i];chars[i] ^= chars[j];}}
}
复杂度分析
  • 时间复杂度:(O(n)),需要多次遍历字符数组,但总的时间复杂度仍为线性。
  • 空间复杂度:(O(n)),主要用于存储字符数组。

解法四:对 String 的 char [] 数组操作

class Solution {//用 char[] 来实现 String 的 removeExtraSpaces,reverse 操作public String reverseWords(String s) {char[] chars = s.toCharArray();//1.去除首尾以及中间多余空格chars = removeExtraSpaces(chars);//2.整个字符串反转reverse(chars, 0, chars.length - 1);//3.单词反转reverseEachWord(chars);return new String(chars);}//1.用 快慢指针 去除首尾以及中间多余空格,可参考数组元素移除的题解public char[] removeExtraSpaces(char[] chars) {int slow = 0;for (int fast = 0; fast < chars.length; fast++) {//先用 fast 移除所有空格if (chars[fast] != ' ') {//在用 slow 加空格。 除第一个单词外,单词末尾要加空格if (slow != 0)chars[slow++] = ' ';//fast 遇到空格或遍历到字符串末尾,就证明遍历完一个单词了while (fast < chars.length && chars[fast] != ' ')chars[slow++] = chars[fast++];}}//相当于 c++ 里的 resize()char[] newChars = new char[slow];System.arraycopy(chars, 0, newChars, 0, slow);return newChars;}//双指针实现指定范围内字符串反转,可参考字符串反转题解public void reverse(char[] chars, int left, int right) {if (right >= chars.length) {System.out.println("set a wrong right");return;}while (left < right) {chars[left] ^= chars[right];chars[right] ^= chars[left];chars[left] ^= chars[right];left++;right--;}}//3.单词反转public void reverseEachWord(char[] chars) {int start = 0;//end <= s.length() 这里的 = ,是为了让 end 永远指向单词末尾后一个位置,这样 reverse 的实参更好设置for (int end = 0; end <= chars.length; end++) {// end 每次到单词末尾后的空格或串尾,开始反转单词if (end == chars.length || chars[end] == ' ') {reverse(chars, start, end - 1);start = end + 1;}}}
}
复杂度分析
  • 时间复杂度:(O(n)),每个步骤都需要遍历字符数组一次。
  • 空间复杂度:(O(n)),主要用于存储新的字符数组。

总结

以上四种解法都可以解决翻转字符串里单词的问题,时间复杂度均为 (O(n)),空间复杂度均为 (O(n))。解法一和解法四的思路较为相似,都是先处理空格,再进行反转操作;解法二通过创建新数组的方式从后往前填充单词;解法三则采用双反转和移位的策略。在实际应用中,可以根据具体需求和代码风格选择合适的解法。

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com

热搜词