This documentation is automatically generated by online-judge-tools/verification-helper
#define PROBLEM "http://judge.u-aizu.ac.jp/onlinejudge/description.jsp?id=2873&lang=jp"
#include "../memo/macro.hpp"
#include "../string/aho_corasick.cpp"
signed main(void) {
int n;
string s;
cin >> s >> n;
vector<string> p(n);
REP(i, n) cin >> p[i];
AhoCorasick<26> aho(p);
int ret = 0, now = aho.root;
for(auto c: s) {
now = aho.next(now, c);
if(aho.nodes[now].matched.size()) {
ret++;
now = aho.root;
}
}
cout << ret << endl;
return 0;
}
#line 1 "test/aoj2873.test.cpp"
#define PROBLEM "http://judge.u-aizu.ac.jp/onlinejudge/description.jsp?id=2873&lang=jp"
#line 1 "memo/macro.hpp"
#include <bits/stdc++.h>
using namespace std;
using ll = long long;
using PII = pair<ll, ll>;
#define FOR(i, a, n) for (ll i = (ll)a; i < (ll)n; ++i)
#define REP(i, n) FOR(i, 0, n)
#define ALL(x) x.begin(), x.end()
template<typename T> void chmin(T &a, const T &b) { a = min(a, b); }
template<typename T> void chmax(T &a, const T &b) { a = max(a, b); }
struct FastIO {FastIO() { cin.tie(0); ios::sync_with_stdio(0); }}fastiofastio;
const ll INF = 1LL<<60;
#line 1 "string/aho_corasick.cpp"
// BEGIN CUT
// 文字の種類数をtemplate引数で渡す
template <int types = 26>
struct AhoCorasick {
struct node {
int fail;
vector<int> next;
vector<int> matched;
node() : fail(-1), next(types, -1) {}
};
vector<node> nodes;
// 辞書の種類数, trie木の根
int sz, root;
// 文字と数字の対応付けをする関数
using F = function<int(char)>;
F trans;
AhoCorasick() {}
AhoCorasick(vector<string> pattern, F f = [](char c){return c-'a';}) : sz(pattern.size()), root(0) {
nodes.resize(1);
trans = f;
build(pattern);
}
// vectorを結合
vector<int> unite(const vector<int> &a, const vector<int> &b) {
vector<int> ret;
set_union(ALL(a), ALL(b), back_inserter(ret));
return ret;
}
// 文字列集合patternからtrie木っぽいオートマトンを作成
void build(vector<string> pattern) {
int now;
nodes[root].fail = root;
REP(i, pattern.size()) {
now = root;
for(const auto &c: pattern[i]) {
if(nodes[now].next[trans(c)] == -1) {
nodes.push_back(node());
nodes[now].next[trans(c)] = nodes.size() - 1;
}
now = nodes[now].next[trans(c)];
}
nodes[now].matched.push_back(i);
}
queue<int> que;
REP(i, types) {
if(nodes[root].next[i] == -1) {
nodes[root].next[i] = root;
} else {
nodes[nodes[root].next[i]].fail = root;
que.push(nodes[root].next[i]);
}
}
while(que.size()) {
now = que.front(); que.pop();
REP(i, types) {
if(nodes[now].next[i] != -1) {
int nxt = nodes[now].fail;
while(nodes[nxt].next[i] == -1) nxt = nodes[nxt].fail;
int nxt_tmp = nodes[now].next[i];
nodes[nxt_tmp].fail = nodes[nxt].next[i];
nodes[nxt_tmp].matched = unite(nodes[nxt_tmp].matched, nodes[nodes[nxt].next[i]].matched);
que.push(nxt_tmp);
}
}
}
}
// 一文字ずつ照合していく
int next(int p, const char c) {
while(nodes[p].next[trans(c)] == -1) p = nodes[p].fail;
return nodes[p].next[trans(c)];
}
// 文字列s中に辞書と一致する部分列がどれだけあるか
vector<int> match(const string s) {
vector<int> res(sz);
int now = root;
for(auto c : s) {
now = next(now, c);
for(auto i : nodes[now].matched) res[i]++;
}
return res;
}
};
// END CUT
#line 4 "test/aoj2873.test.cpp"
signed main(void) {
int n;
string s;
cin >> s >> n;
vector<string> p(n);
REP(i, n) cin >> p[i];
AhoCorasick<26> aho(p);
int ret = 0, now = aho.root;
for(auto c: s) {
now = aho.next(now, c);
if(aho.nodes[now].matched.size()) {
ret++;
now = aho.root;
}
}
cout << ret << endl;
return 0;
}