1. 首页
  2. 水题

HDU 5672 String (尺取法)

有一个明显的性质:如果子串(i,j)(i,j)(i,j)包含了至少kkk个不同的字符,那么子串(i,k),(j<k<length)(i,k),(j < k < length)(i,k),(j<k<length)也包含了至少kkk个不同字符。

因此对于每一个左边界,只要找到最小的满足条件的右边界,就能在O(1)O(1)O(1)时间内统计完所有以这个左边界开始的符合条件的子串。

寻找这个右边界,是经典的追赶法(尺取法,双指针法)问题。维护两个指针(数组下标),轮流更新左右边界,同时累加答案即可。复杂度 O(length(S))O(length(S))O(length(S))

#include <bits/stdc++.h>
using namespace std;
typedef long long LL;

char s[1000010];
int a[30];
int k;

int main() {
	int t;
	scanf("%d", &t);
	while (t--) {
		scanf("%s", s + 1);
		memset(a, 0, sizeof(a));
		scanf("%d", &k);
		//cout<<(s+1)<<endl;
		int len = strlen(s + 1);
		LL ans = 0;
		int cnt = 0;
		int j = 0;
		for (int i = 1; i <= len; ++i) {
			while (cnt < k && j + 1 <= len) {
				j++;
				a[s[j] - 'a']++;
				if (a[s[j] - 'a'] == 1) {
					cnt++;
				}
			}
			if (cnt == k) {
				ans += len - j + 1;
			}
			a[s[i] - 'a']--;
			if (a[s[i] - 'a'] == 0) {
				cnt--;
			}
		}
		cout << ans << endl;

	}

	return 0;
}

 

评分 0, 满分 5 星
0
0
看完收藏一下,下次也能找得到
  • 版权声明:本文基于《知识共享署名-相同方式共享 3.0 中国大陆许可协议》发布,转载请遵循本协议
  • 文章链接:http://www.carlstedt.cn/archives/864 (转载时请注明本文出处及文章链接)
  • 本文无相关文章
上一篇:
:下一篇

发表评论

gravatar

快来吐槽一下吧!

  1. .01 4:06
  2. .02 1:47
  3. .03 3:39
  4. .04 1:40