- 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
java解法
- 解题思路
- 该问题与之前的解法类似,需要在字符串 mainStr 中找到一个子串,该子串可以通过调整为字符串 subStr 的排列,且允许在子串前后插入至多 extraLen 个额外字符。为此,问题可以通过滑动窗口和字符频率计数器来解决。
具体思路:
初始化字符频率表: 使用一个大小为 128 的数组 freqMap 来记录 subStr 中各字符的需求量。
初始化窗口: 首先处理 mainStr 的前 subLen + extraLen 个字符(初始窗口),检查是否已经满足条件。
动态调整窗口:
滑动窗口时,每次移除窗口起点字符,加入窗口终点字符,动态调整 freqMap 和未匹配字符的计数 missingCount。
检查匹配条件: 当 missingCount == 0 时,表示窗口内的字符可以通过排列形成 subStr,返回当前窗口起始索引。
无匹配返回 -1: 如果所有窗口均不满足条件,则返回 -1。
import java.util.Scanner;
public class Main {
public static void main(String[] args) {
Scanner sc = new Scanner(System.in);
String subStr = sc.next();
String mainStr = sc.next();
int extraLen = sc.nextInt();
System.out.println(findMinIndex(subStr, mainStr, extraLen));
}
public static int findMinIndex(String subStr, String mainStr, int extraLen) {
int subLen = subStr.length();
int mainLen = mainStr.length();
if (mainLen < subLen + extraLen) return -1;
int[] freqMap = new int[128];
for (int i = 0; i < subLen; i++) {
freqMap[subStr.charAt(i)]++;
}
int missingCount = subLen;
int maxIndex = mainLen - subLen - extraLen;
int targetLen = subLen + extraLen;
for (int j = 0; j < targetLen; j++) {
if (--freqMap[mainStr.charAt(j)] >= 0) {
missingCount--;
}
if (missingCount == 0) {
return 0;
}
}
for (int i = 1; i <= maxIndex; i++) {
if (++freqMap[mainStr.charAt(i - 1)] > 0) {
missingCount++;
}
if (--freqMap[mainStr.charAt(i + targetLen - 1)] >= 0) {
missingCount--;
}
if (missingCount == 0) {
return i;
}
}
return -1;
}
}
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
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
C++解法
- 解题思路
- 该问题要求在字符串 s2 中找到一个子串,使其包含字符串 s1 的所有字符,且允许在前后插入最多 k 个额外字符。为此,可以通过滑动窗口和字符频率计数的方式解决。
具体思路:
初始化字符频率表: 使用两个数组 cnt1 和 win 分别记录 s1 的字符频率和当前窗口的字符频率。
匹配计数器: 使用变量 match 记录窗口中与 s1 对应字符频率匹配的字符数量。
窗口初始化: 窗口长度为 len1 + k,初始时将 s2 的前 len1 + k 个字符加入窗口。
滑动窗口: 每次移动窗口:
加入新的字符,更新窗口频率,并检查是否增加匹配字符数量。
移除旧字符,更新窗口频率,并检查是否减少匹配字符数量。
匹配判断: 当 match == len1 时,表示窗口中字符完全覆盖了 s1 的所有字符,返回当前窗口的起始索引。
无匹配返回: 若遍历所有窗口后仍未找到匹配,返回 -1。
#include
#include
#include
using namespace std;
int findSubstr(string s1, string s2, int k) {
int len1 = s1.size();
int len2 = s2.size();
if (len2 < len1 + k) return -1;
vector<int> cnt1(128, 0), win(128, 0);
for (char c : s1) {
cnt1[c]++;
}
int match = 0;
int winLen = len1 + k;
for (int i = 0; i < len2; i++) {
if (win[s2[i]]++ < cnt1[s2[i]]) match++;
if (i >= winLen) {
if (--win[s2[i - winLen]] < cnt1[s2[i - winLen]]) match--;
}
if (match == len1) return i - winLen + 1;
}
return -1;
}
int main() {
string s1, s2;
int k;
cin >> s1 >> s2 >> k;
cout << findSubstr(s1, s2, k) << 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
- 52
- 53
- 54
- 55
- 56
C解法
更新中
class="hljs-button signin active" data-title="登录复制" data-report-click="{"spm":"1001.2101.3001.4334"}">
JS解法
初始化字符计数器:
使用哈希表 count 记录字符串 s1 中每个字符的需求频率。
窗口匹配计数器:
使用变量 missing 表示当前窗口中未匹配的 s1 中的字符数量。
初始化窗口:
窗口长度为 len1 + k,初始时将 s2 的前 len1 + k 个字符加入窗口,并动态更新 missing。
滑动窗口:
每次滑动窗口时,移出窗口的第一个字符,加入窗口的最后一个字符,同时更新 count 和 missing。
检查匹配条件:
当 missing == 0 时,表示窗口内字符完全覆盖了 s1 的所有字符,返回当前窗口的起始索引。
无匹配返回:
若遍历所有窗口后未找到匹配,返回 -1。
const readline = require("readline");
const rl = readline.createInterface({
input: process.stdin,
output: process.stdout,
});
const input = [];
rl.on("line", (line) => {
input.push(line);
if (input.length === 3) {
const [s1, s2, k] = input;
console.log(findMinIndex(s1, s2, parseInt(k)));
input.length = 0;
}
});
function findMinIndex(s1, s2, k) {
const len1 = s1.length;
const len2 = s2.length;
if (len2 < len1 + k) return -1;
const count = {};
for (let ch of s1) {
count[ch] = (count[ch] || 0) + 1;
}
let missing = len1;
const windowSize = len1 + k;
for (let i = 0; i < windowSize; i++) {
const ch = s2[i];
if (count[ch] !== undefined) {
if (count[ch] > 0) missing--;
count[ch]--;
}
if (missing === 0) return 0;
}
for (let i = 1; i <= len2 - windowSize; i++) {
const toRemove = s2[i - 1];
if (count[toRemove] !== undefined) {
if (count[toRemove] >= 0) missing++;
count[toRemove]++;
}
const toAdd = s2[i - 1 + windowSize];
if (count[toAdd] !== undefined) {
if (count[toAdd] > 0) missing--;
count[toAdd]--;
}
if (missing === 0) return i;
}
return -1;
}
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
- 59
- 60
- 61
- 62
- 63
- 64
- 65
- 66
- 67
- 68
- 69
- 70
- 71
注意:
如果发现代码有用例覆盖不到的情况,欢迎反馈!会在第一时间修正,更新。
解题不易,如对您有帮助,欢迎点赞/收藏
评论记录:
回复评论: