一、问题描述
题目描述
输入单行英文句子,里面包含英文字母,空格以及,.?三种标点符号,请将句子内每个单词进行倒序,并输出倒序后的语句。
输入描述
输入字符串S,S的长度 1 ≤ N ≤ 100
输出描述
输出倒序后的字符串
备注
标点符号左右的空格 ≥ 0,单词间空格>0
用例
输入
yM eman si boB.
输出
My name is Bob.
说明
无
输入
woh era uoy ? I ma enif.
输出
how are you ? I am fine.
说明
无
题目解析
从用例可以看出,单词的倒序并不难,将字符串单词转为字符数组后,reverse一下就行了。但是单词中如果有标点符号的话,则标点符号的位置不能改变,比如enif. 倒序后为 fine. 其中 . 的位置在倒序前后是一样的。
我的解题思路
从左到右遍历每一个字符,如果字符是 , . ? 或者空格,则看成一个分界符,将分界符之间的单词片段进行倒序。
二、JavaScript算法源码
以下是基于 JavaScript 的代码的中文详细注释和逻辑讲解:
代码逻辑
/* JavaScript Node ACM模式 控制台输入获取 */
const readline = require("readline");// 创建 readline 接口,用于从控制台读取输入
const rl = readline.createInterface({input: process.stdin, // 输入流output: process.stdout, // 输出流
});// 监听输入事件,每次输入一行时触发
rl.on("line", (line) => {// 调用 getResult 函数处理输入,并输出结果console.log(getResult(line));
});// 处理输入字符串的函数
function getResult(str) {// 定义正则表达式,匹配逗号、句号、问号和空格const reg = /[\,\.\?\s]/;// 定义数组 idxs,用于存储分隔符的位置// 初始值为 -1,表示字符串的起始位置const idxs = [-1];// 遍历输入字符串,找到所有分隔符的位置for (let i = 0; i < str.length; i++) {if (reg.test(str[i])) {idxs.push(i); // 将分隔符的位置存入 idxs}}// 将字符串的结束位置(str.length)加入 idxsidxs.push(str.length);// 将字符串转换为数组,方便修改const arr = [...str];// 使用 reduce 方法遍历 idxs 数组// 对每个分隔符之间的子串进行反转idxs.reduce((p, c) => {// p 是前一个分隔符的位置,c 是当前分隔符的位置let l = p + 1; // 子串的起始位置let r = c - 1; // 子串的结束位置// 反转子串while (l < r) {let tmp = arr[l];arr[l] = arr[r];arr[r] = tmp;l++;r--;}// 返回当前分隔符的位置,作为下一次 reduce 的 preturn c;});// 将数组重新拼接为字符串并返回return arr.join("");
}
代码讲解
-
输入处理:
- 使用
readline
模块从控制台读取输入。 - 每次输入一行时,触发
line
事件,调用getResult
函数处理输入。
- 使用
-
分隔符匹配:
- 定义正则表达式
reg
,用于匹配逗号、句号、问号和空格。 - 遍历输入字符串,找到所有分隔符的位置,并存入数组
idxs
。
- 定义正则表达式
-
字符串反转:
- 将输入字符串转换为数组
arr
,方便修改。 - 使用
reduce
方法遍历idxs
数组,对每个分隔符之间的子串进行反转。 - 反转逻辑:
- 定义子串的起始位置
l
和结束位置r
。 - 使用双指针法交换字符,直到
l
和r
相遇。
- 定义子串的起始位置
- 将输入字符串转换为数组
-
结果输出:
- 将反转后的数组
arr
重新拼接为字符串并返回。
- 将反转后的数组
示例解析
输入
Hello, world! This is a test.
运行结果
olleH, dlrow! sihT si a tset.
- 解析:
- 分隔符包括逗号、空格和句号。
- 反转每个分隔符之间的子串:
Hello
→olleH
world
→dlrow
This
→sihT
is
→si
a
→a
test
→tset
总结
- 该代码通过正则表达式匹配分隔符,并使用双指针法反转每个分隔符之间的子串。
- 适用于需要对字符串中特定分隔符之间的子串进行反转的场景。
- 代码逻辑清晰,注释详细,便于理解和扩展。
更精简的解法,可以利用String.prototype.repalce的正则匹配出输入字符串中各个英文子串,将这些英文子串替换为倒序子串,关于repalce的正则匹配用法请看:
String.prototype.repalce的正则匹配
/* JavaScript Node ACM模式 控制台输入获取 */
const readline = require("readline");const rl = readline.createInterface({input: process.stdin,output: process.stdout,
});rl.on("line", (line) => {console.log(getResult(line));
});function getResult(str) {return str.replace(/[a-zA-Z]+/g, (s) => [...s].reverse().join(""));
}
如果有其他问题,欢迎随时提问!
三、Java算法源码
以下是基于 Java 的代码的中文详细注释和逻辑讲解:
代码逻辑
import java.util.ArrayList;
import java.util.Scanner;public class Main {public static void main(String[] args) {// 创建 Scanner 对象,用于从控制台读取输入Scanner sc = new Scanner(System.in);// 读取一行输入String str = sc.nextLine();// 调用 getResult 方法处理输入,并输出结果System.out.println(getResult(str));}// 处理输入字符串的方法public static String getResult(String str) {// 定义 ArrayList,用于存储分隔符的位置ArrayList<Integer> idxs = new ArrayList<>();// 初始值为 -1,表示字符串的起始位置idxs.add(-1);// 遍历输入字符串,找到所有分隔符的位置for (int i = 0; i < str.length(); i++) {// 判断当前字符是否为分隔符(逗号、句号、问号或空格)if (",.? ".indexOf(str.charAt(i)) != -1) {idxs.add(i); // 将分隔符的位置存入 idxs}}// 将字符串的结束位置(str.length())加入 idxsidxs.add(str.length());// 将字符串转换为字符数组,方便修改char[] chars = str.toCharArray();// 遍历 idxs 数组,对每个分隔符之间的子串进行反转for (int i = 0; i < idxs.size() - 1; i++) {// 获取当前分隔符之间的子串的起始位置和结束位置int l = idxs.get(i) + 1; // 子串的起始位置int r = idxs.get(i + 1) - 1; // 子串的结束位置// 使用双指针法反转子串while (l < r) {char tmp = chars[l];chars[l] = chars[r];chars[r] = tmp;l++;r--;}}// 将字符数组重新拼接为字符串StringBuilder sb = new StringBuilder();for (char c : chars) {sb.append(c);}// 返回结果字符串return sb.toString();}
}
代码讲解
-
输入处理:
- 使用
Scanner
从控制台读取一行输入。 - 调用
getResult
方法处理输入字符串。
- 使用
-
分隔符匹配:
- 定义
ArrayList<Integer>
存储分隔符的位置。 - 初始值为
-1
,表示字符串的起始位置。 - 遍历输入字符串,找到所有分隔符(逗号、句号、问号或空格)的位置,并存入
idxs
。
- 定义
-
字符串反转:
- 将输入字符串转换为字符数组
chars
,方便修改。 - 遍历
idxs
数组,对每个分隔符之间的子串进行反转。 - 反转逻辑:
- 定义子串的起始位置
l
和结束位置r
。 - 使用双指针法交换字符,直到
l
和r
相遇。
- 定义子串的起始位置
- 将输入字符串转换为字符数组
-
结果输出:
- 使用
StringBuilder
将字符数组重新拼接为字符串。 - 返回结果字符串。
- 使用
示例解析
输入
Hello, world! This is a test.
运行结果
olleH, dlrow! sihT si a tset.
- 解析:
- 分隔符包括逗号、空格和句号。
- 反转每个分隔符之间的子串:
Hello
→olleH
world
→dlrow
This
→sihT
is
→si
a
→a
test
→tset
总结
- 该代码通过遍历字符串找到分隔符的位置,并使用双指针法反转每个分隔符之间的子串。
- 适用于需要对字符串中特定分隔符之间的子串进行反转的场景。
- 代码逻辑清晰,注释详细,便于理解和扩展。
如果有其他问题,欢迎随时提问!
四、Python算法源码
以下是基于 Python 的代码的中文详细注释和逻辑讲解:
代码逻辑
原始代码
import re# 从控制台读取输入
s = input()# 定义正则表达式,匹配逗号、句号、问号和空格
p = re.compile(r"[\\,\\.\\?\s]")# 定义数组 idxs,用于存储分隔符的位置
# 初始值为 -1,表示字符串的起始位置
idxs = [-1]# 遍历输入字符串,找到所有分隔符的位置
for i in range(len(s)):if p.match(s[i]):idxs.append(i)# 将字符串的结束位置(len(s))加入 idxs
idxs.append(len(s))# 将字符串转换为列表,方便修改
arr = [c for c in s]# 遍历 idxs 数组,对每个分隔符之间的子串进行反转
for i in range(len(idxs) - 1):l = idxs[i] + 1 # 子串的起始位置r = idxs[i + 1] - 1 # 子串的结束位置# 使用双指针法反转子串while l < r:arr[l], arr[r] = arr[r], arr[l]l += 1r -= 1# 将列表重新拼接为字符串并输出
print("".join(arr))
优化后的代码
import re# 从控制台读取输入
s = input()# 定义替换函数 rep,用于反转匹配的子串
def rep(matched):# 将匹配的子串转换为列表并反转tmp = list(matched.group())tmp.reverse()# 将反转后的列表重新拼接为字符串并返回return "".join(tmp)# 使用 re.sub 方法,将匹配的子串替换为反转后的结果
# 正则表达式 r"[a-zA-Z]+" 匹配一个或多个字母
print(re.sub(r"[a-zA-Z]+", rep, s))
代码讲解
原始代码
-
输入处理:
- 使用
input()
从控制台读取输入字符串。
- 使用
-
分隔符匹配:
- 定义正则表达式
p
,用于匹配逗号、句号、问号和空格。 - 遍历输入字符串,找到所有分隔符的位置,并存入数组
idxs
。
- 定义正则表达式
-
字符串反转:
- 将输入字符串转换为列表
arr
,方便修改。 - 遍历
idxs
数组,对每个分隔符之间的子串进行反转。 - 反转逻辑:
- 定义子串的起始位置
l
和结束位置r
。 - 使用双指针法交换字符,直到
l
和r
相遇。
- 定义子串的起始位置
- 将输入字符串转换为列表
-
结果输出:
- 将反转后的列表
arr
重新拼接为字符串并输出。
- 将反转后的列表
优化后的代码
-
输入处理:
- 使用
input()
从控制台读取输入字符串。
- 使用
-
替换函数:
- 定义函数
rep
,用于处理正则匹配的结果。 - 将匹配的子串转换为列表并反转,然后重新拼接为字符串。
- 定义函数
-
正则替换:
- 使用
re.sub
方法,将匹配的子串替换为反转后的结果。 - 正则表达式
r"[a-zA-Z]+"
匹配一个或多个字母。
- 使用
-
结果输出:
- 直接输出替换后的字符串。
示例解析
输入
Hello, world! This is a test.
原始代码运行结果
olleH, dlrow! sihT si a tset.
- 解析:
- 分隔符包括逗号、空格和句号。
- 反转每个分隔符之间的子串:
Hello
→olleH
world
→dlrow
This
→sihT
is
→si
a
→a
test
→tset
优化后代码运行结果
olleH, dlrow! sihT si a tset.
- 解析:
- 正则表达式
r"[a-zA-Z]+"
匹配所有字母子串。 - 对每个匹配的子串进行反转:
Hello
→olleH
world
→dlrow
This
→sihT
is
→si
a
→a
test
→tset
- 正则表达式
总结
- 原始代码通过遍历字符串找到分隔符的位置,并使用双指针法反转每个分隔符之间的子串。
- 优化后的代码利用
re.sub
方法和正则表达式,直接匹配字母子串并反转,代码更简洁高效。 - 两种方法均适用于需要对字符串中特定子串进行反转的场景。
如果有其他问题,欢迎随时提问!
五、C/C++算法源码:
以下是基于 C++ 的代码实现,并附上中文详细注释和逻辑讲解:
C++ 代码实现
#include <iostream>
#include <string>
#include <vector>
#include <algorithm>
#include <regex>using namespace std;// 定义替换函数 rep,用于反转匹配的子串
string rep(const smatch& matched) {string tmp = matched.str(); // 获取匹配的子串reverse(tmp.begin(), tmp.end()); // 反转子串return tmp; // 返回反转后的子串
}int main() {// 从控制台读取输入string s;getline(cin, s);// 定义正则表达式,匹配一个或多个字母regex pattern("[a-zA-Z]+");// 使用 regex_replace 方法,将匹配的子串替换为反转后的结果// 第三个参数 rep 是一个函数,用于处理匹配的子串string result = regex_replace(s, pattern, rep);// 输出结果cout << result << endl;return 0;
}
代码讲解
1. 输入处理
- 使用
getline(cin, s)
从控制台读取一行输入,并存储到字符串s
中。
2. 正则表达式
- 定义正则表达式
pattern
,用于匹配一个或多个字母([a-zA-Z]+
)。
3. 替换函数
- 定义函数
rep
,用于处理正则匹配的结果。- 参数
matched
是smatch
类型,表示正则匹配的结果。 - 使用
matched.str()
获取匹配的子串。 - 使用
reverse(tmp.begin(), tmp.end())
反转子串。 - 返回反转后的子串。
- 参数
4. 正则替换
- 使用
regex_replace
方法,将输入字符串s
中所有匹配的子串替换为反转后的结果。- 第一个参数是输入字符串
s
。 - 第二个参数是正则表达式
pattern
。 - 第三个参数是替换函数
rep
。
- 第一个参数是输入字符串
5. 结果输出
- 输出替换后的字符串
result
。
示例解析
输入
Hello, world! This is a test.
运行结果
olleH, dlrow! sihT si a tset.
- 解析:
- 正则表达式
[a-zA-Z]+
匹配所有字母子串。 - 对每个匹配的子串进行反转:
Hello
→olleH
world
→dlrow
This
→sihT
is
→si
a
→a
test
→tset
- 正则表达式
总结
- 该代码利用 C++ 的正则表达式库
<regex>
,实现了对输入字符串中所有字母子串的反转。 - 通过
regex_replace
方法和自定义替换函数rep
,代码简洁高效。 - 适用于需要对字符串中特定子串进行反转的场景。
如果有其他问题,欢迎随时提问!