错题-巨石迷阵

巨石迷阵

问题描述

听说这片土地埋藏着什么秘密,来到这片土地的人不计其数,有人说这里财宝无数,也有人说这里是上古文明留下来的遗迹。小 L 收集情报和资料很久了,只身一人历经千辛万苦终于来到了这片地域的中心地带。突然,四周升起许多巨石,不出所料,面前的正是巨石迷阵。

你面前有 n 块巨石排成一行,每个上面有一个大写字母。
接下来有 m 个询问,每一个询问包含两个数字 l,r ,对于每个询问,你需要回答这个处于区间 [l,r] 的石块上的字母是否每一个英文字母都至少出现了一次。

输入格式

第一行一个整数 n , n≤5×105
第二行,一个长度为 n 的字符串
第三行,一个整数 m, m≤5×105
接下来的 m 行,每行两个整数表示 l,r,1≤lrn

输出格式

输出包含 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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值