究极反转
344.反转字符串
题目链接: leetcode.cn/problems/re…
题目要求:编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 s 的形式给出。
不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。
代码:
class Solution {
public void reverseString(char[] s) {
int left = 0;
int right = s.length - 1;
while (left < right) {
char temp = s[left];
s[left] = s[right];
s[right] = temp;
left++;
right--;
}
}
}
解题之思考及结果
学习到了双指针的新用法,成功AC
541. 反转字符串 II
题目要求:给定一个字符串 s 和一个整数 k,从字符串开头算起,每计数至 2k 个字符,就反转这 2k 字符中的前 k 个字符。
如果剩余字符少于 k 个,则将剩余字符全部反转。 如果剩余字符小于 2k 但大于或等于 k 个,则反转前 k 个字符,其余字符保持原样。
代码:
class Solution {
public String reverseStr(String s, int k) {
char[] c = s.toCharArray();
int count = 0;
// left、right维持待反转字符串下标
int left = 0;
int right = 0;
for (int i = 0; i < c.length; i++) {
if (++count == 2 * k) {
while (left < right) {
char temp = c[left];
c[left] = c[right];
c[right] = temp;
left++;
right--;
}
left = i + 1;
right = left;
count = 0;
}
// 让右指针保持在前k个字符上
right += right > k + left - 2 ? 0 : 1;
}
// 收尾工作,反转剩余字符
if (right >= c.length) {
right = c.length - 1;
}
while (left < right) {
char temp = c[left];
c[left] = c[right];
c[right] = temp;
left++;
right--;
}
return String.valueOf(c);
}
}
解题之思考及结果
这题也是自己可以想出来的题,采用了类似滑动窗口的方法,两个指针维持了待反转的范围
剑指Offer 05.替换空格
题目要求:请实现一个函数,把字符串 s 中的每个空格替换成"%20"。
代码:
class Solution {
public String replaceSpace(String s) {
StringBuilder res = new StringBuilder();
for (char i : s.toCharArray()) {
if (i == ' ') {
res.append("%20");
continue;
}
res.append(i);
}
return res.toString();
}
}
解题之思考及结果
暴力解挺容易的,难点在空间复杂度O(1)。
可以扩充数组到每个空格替换成"%20"之后的大小。然后从后向前替换空格,也就是双指针法
151.翻转字符串里的单词
题目要求:给你一个字符串 s ,请你反转字符串中 单词 的顺序。
单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开。
返回 单词 顺序颠倒且 单词 之间用单个空格连接的结果字符串。
注意:输入字符串 s中可能会存在前导空格、尾随空格或者单词间的多个空格。返回的结果字符串中,单词间应当仅用单个空格分隔,且不包含任何额外的空格。
代码:
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--;
}
// System.out.println("ReverseWords.reverseString returned: sb = [" + sb + "]");
}
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;
}
}
}
解题之思考及结果
还没成功领悟
剑指Offer58-II.左旋转字符串
题目要求:字符串的左旋转操作是把字符串前面的若干个字符转移到字符串的尾部。请定义一个函数实现字符串左旋转操作的功能。比如,输入字符串"abcdefg"和数字2,该函数将返回左旋转两位得到的结果"cdefgab"。
代码:
class Solution {
public String reverseLeftWords(String s, int n) {
char[] c = s.toCharArray();
char temp;
while (n-- > 0){
int i = 0;
temp = c[0];
while (i < s.length() - 1) {
c[i] = c[i+1];
i++;
}
c[c.length - 1] = temp;
}
return String.valueOf(c);
}
}
解题之思考及结果
更快的做法是再次利用反转的特性,将前n个字符反转,再将n后面的字符串反转,最后整个字符串反转