結果
| 問題 |
No.430 文字列検索
|
| コンテスト | |
| ユーザー |
ei1333333
|
| 提出日時 | 2016-10-09 17:54:57 |
| 言語 | C++11(廃止可能性あり) (gcc 13.3.0) |
| 結果 |
AC
|
| 実行時間 | 14 ms / 2,000 ms |
| コード長 | 2,415 bytes |
| コンパイル時間 | 1,234 ms |
| コンパイル使用メモリ | 166,240 KB |
| 実行使用メモリ | 5,248 KB |
| 最終ジャッジ日時 | 2024-11-10 00:10:13 |
| 合計ジャッジ時間 | 1,933 ms |
|
ジャッジサーバーID (参考情報) |
judge5 / judge3 |
(要ログイン)
| ファイルパターン | 結果 |
|---|---|
| sample | AC * 4 |
| other | AC * 14 |
ソースコード
#include <bits/stdc++.h>
using namespace std;
struct SuffixArray
{
vector< int > SA, LCP;
string s;
void Build_SA(const string& str)
{
s = str;
SA.resize(s.size());
iota(begin(SA), end(SA), 0);
sort(begin(SA), end(SA), [&](const int& a, const int& b)
{
if(s[a] == s[b]) return (a > b);
return (s[a] < s[b]);
});
vector< int > classes(s.size()), c(s.size()), cnt(s.size());
for(int i = 0; i < s.size(); i++) {
c[i] = s[i];
}
for(int len = 1; len < s.size(); len <<= 1) {
for(int i = 0; i < s.size(); i++) {
if(i > 0 && c[SA[i - 1]] == c[SA[i]] && SA[i - 1] + len < s.size() && c[SA[i - 1] + len / 2] == c[SA[i] + len / 2]) {
classes[SA[i]] = classes[SA[i - 1]];
} else {
classes[SA[i]] = i;
}
}
iota(begin(cnt), end(cnt), 0);
copy(begin(SA), end(SA), begin(c));
for(int i = 0; i < s.size(); i++) {
int s1 = c[i] - len;
if(s1 >= 0) SA[cnt[classes[s1]]++] = s1;
}
classes.swap(c);
}
}
void Build_LCP()
{
vector< int > rank(s.size());
for(int i = 0; i < s.size(); i++) {
rank[SA[i]] = i;
}
LCP.resize(s.size() - 1);
for(int i = 0, h = 0; i < s.size(); i++) {
if(rank[i] + 1 < s.size()) {
for(int j = SA[rank[i] + 1]; max(i, j) + h < s.length() && s[i + h] == s[j + h]; ++h);
LCP[rank[i]] = h;
if(h > 0) --h;
}
}
}
int operator[](int k) const
{
return (SA[k]);
}
int size() const
{
return (s.size());
}
bool lt_substr(string& t, int si = 0, int ti = 0)
{
int sn = s.size(), tn = t.size();
while(si < sn && ti < tn) {
if(s[si] < t[ti]) return (true);
if(s[si] > t[ti]) return (false);
++si, ++ti;
}
return (si >= sn && ti < tn);
}
int lower_bound(string& t, int low = -1)
{
int high = SA.size();
while(high - low > 1) {
int mid = (low + high) >> 1;
if(lt_substr(t, SA[mid])) low = mid;
else high = mid;
}
return (high);
}
};
int main()
{
cin.tie(0);
ios::sync_with_stdio(false);
string S;
cin >> S;
SuffixArray sa;
sa.Build_SA(S);
int M;
cin >> M;
int ret = 0;
while(M--) {
string T;
cin >> T;
int up = sa.lower_bound(T);
T.back()++;
int down = sa.lower_bound(T, up - 1);
ret += down - up;
}
cout << ret << endl;
}
ei1333333