- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
java解法
- 解题思路
- 解题思路
这道题的目的是在给定的字符串 str2 中找到一个子串,该子串是字符串 str1 的字母排列之一(即 str1 中字符的重排)。如果找到,返回第一个符合条件的子串的起始索引;如果找不到,返回 -1。
为了解决这个问题,我们可以使用 滑动窗口 和 字符计数 的方法。具体步骤如下:
初始化字符计数:首先计算 str1 中每个字符的频率,存储在 targetCount 数组中。然后,使用一个滑动窗口遍历 str2,在窗口中计算字符的频率,存储在 windowCount 数组中。
初始窗口匹配:首先,检查 str2 的前 n 个字符(n 是 str1 的长度)是否与 str1 的字符频率匹配。如果匹配,返回起始索引 0。
滑动窗口:将窗口从 str2 的前 n 个字符滑动到后面的字符。每次滑动窗口时,更新 windowCount,即移除窗口最左边的字符并加入窗口右边的新字符。每次更新后,检查窗口中的字符频率是否与 targetCount 匹配,如果匹配,返回当前窗口的起始位置。
结束条件:如果遍历完所有窗口仍未找到匹配的子串,返回 -1。
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner scanner = new Scanner(System.in);
String str1 = scanner.next();
String str2 = scanner.next();
System.out.println(findSubstr(str1, str2));
}
public static int findSubstr(String str1, String str2) {
int[] targetCount = new int[26];
int[] windowCount = new int[26];
for (char c : str1.toCharArray()) {
targetCount[c - 'a']++;
}
for (int i = 0; i < str1.length(); i++) {
windowCount[str2.charAt(i) - 'a']++;
}
if (matches(targetCount, windowCount)) {
return 0;
}
for (int i = str1.length(); i < str2.length(); i++) {
windowCount[str2.charAt(i - str1.length()) - 'a']--;
windowCount[str2.charAt(i) - 'a']++;
if (matches(targetCount, windowCount)) {
return i - str1.length() + 1;
}
}
return -1;
}
private static boolean matches(int[] targetCount, int[] windowCount) {
for (int i = 0; i < 26; i++) {
if (targetCount[i] != windowCount[i]) {
return false;
}
}
return true;
}
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
- 58
C++解法
- 解题思路
- 解题思路
这道题的目的是在字符串 str2 中找到一个与 str1 字符排列相同的子串(即 str1 中字符的重排)。我们可以利用 滑动窗口 和 字符计数 技巧来高效地解决此问题。具体的思路如下:
字符计数:首先统计 str1 中每个字符的频率,存储在 targetCount 数组中。接着使用滑动窗口的方法遍历 str2,统计窗口中的字符频率,并存储在 windowCount 数组中。
初始窗口匹配:首先,检查 str2 的前 n 个字符(n 为 str1 的长度)是否与 str1 的字符频率匹配。如果匹配,直接返回起始索引 0。
滑动窗口:将窗口从 str2 的前 n 个字符滑动到后面的字符。每次滑动时,更新 windowCount 数组(即移除窗口最左边的字符并加入新字符)。每次更新后,检查窗口中的字符频率是否与 str1 的频率相匹配。如果匹配,返回当前窗口的起始位置。
结束条件:如果遍历完所有窗口仍未找到匹配的子串,则返回 -1。
#include
#include
#include
using namespace std;
int findSubstr(const string& str1, const string& str2) {
vector<int> targetCount(26, 0);
vector<int> windowCount(26, 0);
for (char c : str1) {
targetCount[c - 'a']++;
}
for (int i = 0; i < str1.length(); i++) {
windowCount[str2[i] - 'a']++;
}
if (targetCount == windowCount) {
return 0;
}
for (int i = str1.length(); i < str2.length(); i++) {
windowCount[str2[i - str1.length()] - 'a']--;
windowCount[str2[i] - 'a']++;
if (targetCount == windowCount) {
return i - str1.length() + 1;
}
}
return -1;
}
int main() {
string str1, str2;
cin >> str1 >> str2;
cout << findSubstr(str1, str2) << endl;
return 0;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
C解法
步骤解析:
字符计数:
我们将 str1 中的每个字符的频率存储在 count1 数组中。
同样,我们将 str2 中当前窗口的字符频率存储在 count2 数组中。
初始窗口:
初始化滑动窗口为 str2 中的前 len1 个字符(即 str1 的长度),并将其字符频率存储在 count2 中。
滑动窗口:
从 str2 的第一个字符开始,滑动窗口每次向右移动一个字符。每移动一步:
移除窗口最左边的字符(即 s2[i-1]),并添加新的字符(即 s2[i + len1 - 1])。
比较 count1 和 count2 是否相等,如果相等,说明当前窗口是 str1 的字母排列,返回当前窗口的起始位置。
结束条件:
如果找到了符合条件的子串,则返回其起始索引;如果遍历整个 str2 后都未找到,则返回 -1。
#include
#include
#define ALPHABET_SIZE 26
int checkInclusion(char* s1, char* s2) {
int len1 = strlen(s1), len2 = strlen(s2);
if (len1 > len2) return -1;
int count1[ALPHABET_SIZE] = {0}, count2[ALPHABET_SIZE] = {0};
for (int i = 0; i < len1; i++) {
count1[s1[i] - 'a']++;
count2[s2[i] - 'a']++;
}
for (int i = 0; i <= len2 - len1; i++) {
if (i > 0) {
count2[s2[i - 1] - 'a']--;
count2[s2[i + len1 - 1] - 'a']++;
}
if (memcmp(count1, count2, sizeof(count1)) == 0) {
return i;
}
}
return -1;
}
int main() {
char str1[100001], str2[100001];
scanf("%s %s", str1, str2);
printf("%d\n", checkInclusion(str1, str2));
return 0;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
JS解法
思路概述:
字符计数:
通过两个字符计数数组,count1 用于统计 str1 中每个字符的频率,count2 用于统计 str2 中当前滑动窗口内每个字符的频率。
初始窗口:
初始化滑动窗口为 str2 中的前 len1 个字符(即 str1 的长度),并将其字符频率存储在 count2 数组中。
滑动窗口:
从 str2 的第一个字符开始,滑动窗口每次向右移动一个字符。每移动一步:
移除窗口最左边的字符(即 s2[i - str1.length]),并添加新的字符(即 s2[i])。
比较 count1 和 count2 是否相等,如果相等,说明当前窗口是 str1 的字母排列,返回当前窗口的起始位置。
结束条件:
如果找到了符合条件的子串,则返回其起始索引;如果遍历整个 str2 后都未找到,则返回 -1。
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
rl.on("line", (line) => {
let [str1, str2] = line.split(" ");
console.log(findSubstring(str1, str2));
});
function findSubstring(str1, str2) {
const count1 = Array(26).fill(0);
const count2 = Array(26).fill(0);
for (let i = 0; i < str1.length; i++) {
count1[str1.charCodeAt(i) - 97]++;
}
for (let i = 0; i < str1.length; i++) {
count2[str2.charCodeAt(i) - 97]++;
}
if (arraysEqual(count1, count2)) return 0;
for (let i = str1.length; i < str2.length; i++) {
count2[str2.charCodeAt(i - str1.length) - 97]--;
count2[str2.charCodeAt(i) - 97]++;
if (arraysEqual(count1, count2)) return i - str1.length + 1;
}
return -1;
}
function arraysEqual(arr1, arr2) {
for (let i = 0; i < 26; i++) {
if (arr1[i] !== arr2[i]) return false;
}
return true;
}
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
- 25
- 26
- 27
- 28
- 29
- 30
- 31
- 32
- 33
- 34
- 35
- 36
- 37
- 38
- 39
- 40
- 41
- 42
- 43
- 44
- 45
- 46
- 47
- 48
- 49
- 50
- 51
- 52
- 53
- 54
- 55
- 56
- 57
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: