题目描述
维护一个字符串集合,支持两种操作:
I x
向集合中插入一个字符串 x;Q x
询问一个字符串在集合中出现了多少次。共有 N 个操作,输入的字符串总长度不超过 10^5,字符串仅包含小写英文字母。
输入格式
第一行包含整数 N,表示操作数。
接下来 N 行,每行包含一个操作指令,指令为
I x
或Q x
中的一种。输出格式
对于每个询问指令
Q x
,都要输出一个整数作为结果,表示 x 在集合中出现的次数。每个结果占一行。
数据范围
1≤N≤2∗10^4
输入样例:
5 I abc Q abc Q ab I ab Q ab
输出样例:
1 0 1
代码
#include<bits/stdc++.h>
using namespace std;
const int N = 2e4+10;
int son[N][26], cnt[N];
int idx = 0;
void insert(std::string& s){
int p = 0;
for(int i = 0; i < s.size(); ++i) {
int u = s[i] - 'a';
if(!son[p][u]) {
son[p][u] = ++idx; // p从0开始, idx不能从0开始
}
p = son[p][u];
}
cnt[p]++; // 统计以p结尾的字符串的个数
}
int query(string& s) {
int p = 0;
for(int i = 0; i < s.size(); ++i) {
int u = s[i] - 'a';
if(!son[p][u]) {
return 0;
} else {
p = son[p][u];
}
}
return cnt[p];
}
int main(){
int m;
cin >> m;
while(m--) {
string op;
cin >> op;
if(op == "I") {
string x;
cin >> x;
insert(x);
} else if(op == "Q") {
string x;
cin >> x;
cout << query(x) << endl;
}
}
return 0;
}