LeetCode 151
思路
头尾双指针法逆置所有单词
- 将字符串的逆转看作是字符串中单词的逆序,因此引入首尾的两个指针交换对应元素
- trim()去除头尾的空格
- 由于空格的长度不固定,此处我使用了正则表达式\\s+来表示任意长度的空格,并以此作为分割基准
- 分割后的数组交换头尾指针对应元素后再拼接在一起
快慢指针法拼接单词
- 从字符串尾部开始,看作是一个单词一个单词的拼接
- trim()去除头尾空格
- 快指针找到第一个为空格的元素,此时说明fast指针刚好经历过了一个单词,那么此时将这个单词拼接在res中,记得加上" "
- 继续移动快指针,直到遇到不为空格的元素,说明此时已经过了单词间的空格情况,将慢指针移动到快指针的位置
注意点
头尾双指针法逆置所有单词
- 只交换到字符串中间的位置,不要重复交换
- 交换后记得移动left,right指针
- 记得拼接" "
快慢指针法拼接单词
- 使用substring的时候要注意它是一个左闭右开的函数,因此范围是[fast+1, slow+1]
- 最后的结果还要去一次头尾的空格
代码
头尾双指针法逆置所有单词
class Solution {
public String reverseWords(String s) {
if (s == null) {
return s;
}
StringBuffer res = new StringBuffer();
s = s.trim(); //去除头尾空格
String[] tmp = s.split("\\s+"); //以空格为基准进行分割
int left = 0;
int right = tmp.length - 1;
for (int i = 0; i < tmp.length; i++) {
if(i < tmp.length / 2) {
swap(left, right, tmp);
}
left++;
right--;
res.append(tmp[i]);
res.append(" ");
}
res.deleteCharAt(res.length()-1);
return res.toString();
}
public static void swap(int left, int right, String[] reTmp) {
String t = reTmp[left];
reTmp[left] = reTmp[right];
reTmp[right] = t;
}
}
快慢指针法拼接单词
class Solution {
public String reverseWords(String s) {
if (s == null) {
return s;
}
StringBuilder res = new StringBuilder();
s = s.trim();
int j = s.length() - 1;
int i = j;
while (i >= 0) {
while (i >= 0 && s.charAt(i) != ' ') //找到第一个空格处
i--;
res.append(s.substring(i + 1, j + 1) + " "); //拼接单词
while (i >= 0 && s.charAt(i) == ' ') //找到不为空格处,删除单词间的空格
i--;
j = i; //j移动到新单词尾部处
}
return res.toString().trim();
}
}
右旋字符串
思路
把输出结果看作是两段字符串的拼接,分割点是字符串的长度-n
代码
import java.util.*;
class Main {
public static void main(String[] args) {
Scanner scan = new Scanner(System.in);
int n = scan.nextInt();
String s = scan.next();
StringBuffer res = new StringBuffer();
int length = s.length();
res.append(s.substring(length-n, length));
res.append(s.substring(0, length-n));
System.out.println(res.toString());
}
}
总结
今天的练习中,对于我来说印象最深刻的就是151的第二种解法,一是这是快慢指针的一个很妙的用法,二是这是一种新的解题想法,除了说可以正向的去思考,还可以逆向思考,就像这道题看作是倒着的单词拼接而不一定要是正向得思考如何变成逆序模样