巨石迷阵
问题描述
听说这片土地埋藏着什么秘密,来到这片土地的人不计其数,有人说这里财宝无数,也有人说这里是上古文明留下来的遗迹。小 L 收集情报和资料很久了,只身一人历经千辛万苦终于来到了这片地域的中心地带。突然,四周升起许多巨石,不出所料,面前的正是巨石迷阵。
你面前有 n 块巨石排成一行,每个上面有一个大写字母。
接下来有 m 个询问,每一个询问包含两个数字 l,r ,对于每个询问,你需要回答这个处于区间 [l,r] 的石块上的字母是否每一个英文字母都至少出现了一次。
输入格式
第一行一个整数 n , n≤5×105
第二行,一个长度为 n 的字符串
第三行,一个整数 m, m≤5×105
接下来的 m 行,每行两个整数表示 l,r,1≤l≤r≤n
输出格式
输出包含 m 行,每行一个 YES ,或者 NO 。
分别表示是否每个字母都至少出现一次。
样例输入
30
AAABCDEFGHIJKLMNOPQRSTUVWXYZAA
5
1 26
2 27
3 28
4 29
5 30
样例输出
NO
NO
YES
YES
NO
题解
使用前缀和
以每一个字母出现的次数作为前缀和计算。在给出区间 l, r 后,就可以使用前缀和计算这个区间中每一个字母是否出现过至少一次。
例如:在字符串中下标为 i 的字母是 26 个子母中第 j 个字母,就把 str[i][j] 置为 1。
#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
ll str[500005][30] = {0};
int main()
{
ll n;
cin >> n;
string s;
cin >> s;
for (ll i = 0; i < n; i++)//在在字符串中下标为 i 的字母是 26 个字母中第 j 个字母,就把 str[i][j] 加上 1,这里为了后面的计算前缀和方便下标都是从1开始的
str[i + 1][s[i] - 'A' + 1]++;
for (ll i = 1; i <= 26; i++)//以每一个字母为单位计算出现次数前缀和
for (ll j = 1; j <= n; j++)
str[j][i] += str[j - 1][i];
ll m;
cin >> m;
for (ll i = 0; i < m; i++)
{
ll l, r;
cin >> l >> r;
bool check = true;
for (ll j = 1; j <= 26; j++)
{
if (str[r][j] - str[l - 1][j] == 0)//当出现这个字母在[l, r]范围中都没有出现过就判断为NO并且可以直接退出循环
{
check = false;
break;
}
}
if (check)
cout << "YES" << endl;
else
cout << "NO" << endl;
}
return 0;
}