LeetCode 344.反转字符串、541.反转字符串II、剑指Offer 05.替换空格、151.翻转字符串里的单词、剑指Offer58-II.左旋转字符串
344. 反转字符串
题目描述
编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 s 的形式给出。
不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。
示例
输入:s = ["h","e","l","l","o"]
输出:["o","l","l","e","h"]
算法实现
public void solution(char[] s) {
int left = 0;
int right = s.length - 1;
while (left < right) {
char temp = s[right];
s[right] = s[left];
s[left] = temp;
left++;
right--;
}
}
541. 反转字符串 II
题目描述
给定一个字符串 s 和一个整数 k,从字符串开头算起,每计数至 2k 个字符,就反转这 2k 字符中的前 k 个字符。
- 如果剩余字符少于 k 个,则将剩余字符全部反转。
- 如果剩余字符小于 2k 但大于或等于 k 个,则反转前 k 个字符,其余字符保持原样。
示例
输入:s = "abcdefg", k = 2
输出:"bacdfeg"
算法实现
public String reverseStr(String s, int k) {
char[] chars = s.toCharArray();
for (int index = 0; index < chars.length; index += 2 * k) {
if (index + k < chars.length) {//防止数组越界 左闭右闭区间
reverse(chars, index, index + k - 1);
continue;
}
reverse(chars, index, chars.length - 1);
}
return new String(chars);
}
public void reverse(char[] s, int left, int right) {
while (left < right) {
char temp = s[right];
s[right] = s[left];
s[left] = temp;
left++;
right--;
}
}
剑指 Offer 05. 替换空格
题目描述
请实现一个函数,把字符串 s 中的每个空格替换成"%20"。
示例
输入:s = "We are happy."
输出:"We%20are%20happy."
算法实现
public String replaceSpace(String s) {
if (null == s || s.length() <= 0) {
return s;
}
StringBuilder builder = new StringBuilder();
char[] chars = s.toCharArray();
for (Character ch : chars) {
if (ch == ' ') {
builder.append("%20");
} else {
builder.append(ch);
}
}
return builder.toString();
}
//双指针法
public String replaceSpace(String s) {
if (null == s || s.length() <= 0) {
return s;
}
StringBuilder builder = new StringBuilder();
char[] chars = s.toCharArray();
for (int index = 0; index < s.length(); index++) {
if (chars[index] == ' ') {
builder.append(" ");
}
}
if (builder.length() == 0) {
//说明该字符串中不存在空格的情况可以直接返回原始的字符串
return s;
}
int left = s.length() - 1;
s += builder;
int right = s.length() - 1;
char[] array = s.toCharArray();
while (left >= 0) {
if (array[left] == ' ') {
array[right--] = '0';
array[right--] = '2';
array[right] = '%';
} else {
array[right] = array[left];
}
left--;
right--;
}
return new String(array);
}
151. 反转字符串中的单词
题目描述
给你一个字符串 s ,请你反转字符串中 单词 的顺序。
单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开。
返回 单词 顺序颠倒且 单词 之间用单个空格连接的结果字符串。
注意:输入字符串 s中可能会存在前导空格、尾随空格或者单词间的多个空格。返回的结果字符串中,单词间应当仅用单个空格分隔,且不包含任何额外的空格。
示例
输入:s = "the sky is blue"
输出:"blue is sky the"
算法实现
public String reverseWords(String s) {
s = s.trim();
// 正则匹配连续的空白字符作为分隔符分割
List<String> wordList = Arrays.asList(s.split("\s+"));
Collections.reverse(wordList);
return String.join(" ", wordList);
}
public String reverseWords(String s) {
StringBuilder sb = removeSpace(s);
reverseString(sb, 0, sb.length() - 1);
reverseEachWord(sb);
return sb.toString();
}
// 1.去除首尾以及中间多余空格
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;
}
// 2.反转整个字符串
private 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--;
}
}
// 3.反转各个单词
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;
}
}
剑指 Offer 58 - II. 左旋转字符串
题目描述
字符串的左旋转操作是把字符串前面的若干个字符转移到字符串的尾部。请定义一个函数实现字符串左旋转操作的功能。比如,输入字符串"abcdefg"和数字2,该函数将返回左旋转两位得到的结果"cdefgab"。
示例
输入: s = "abcdefg", k = 2
输出: "cdefgab"
算法实现
public String reverseLeftWords(String s, int n) {
int length = s.length();
StringBuilder sb = new StringBuilder(s);
reverseString(sb, 0, n - 1);
reverseString(sb, n, length - 1);
return sb.reverse().toString();
}
private void reverseString(StringBuilder sb, int left, int right) {
while (left < right) {
char temp = sb.charAt(left);
sb.setCharAt(left, sb.charAt(right));
sb.setCharAt(right, temp);
left++;
right--;
}
}