This documentation is automatically generated by online-judge-tools/verification-helper
#define PROBLEM "https://judge.yosupo.jp/problem/aho_corasick"
#include "../../string/Trie_Tree.hpp"
#include <string>
#include <iostream>
int main() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
int N;
std::cin >> N;
po167::Trie_Tree<'a', 26> T;
std::vector<int> ans(N);
for (int i = 0; i < N; i++){
std::string S;
std::cin >> S;
auto lis = T.insert(S, i);
ans[i] = lis.back();
}
T.aho();
std::cout << T.nodes.size() << "\n";
for (int i = 1; i < (int)T.nodes.size(); i++){
std::cout << T.nodes[i].parent_node << " " << T.fail[i] << "\n";
}
for (int i = 0; i < N; i++){
std::cout << ans[i] << (i == N - 1 ? "\n" : " ");
}
}
#line 1 "test/string/aho.test.cpp"
#define PROBLEM "https://judge.yosupo.jp/problem/aho_corasick"
#line 2 "string/Trie_Tree.hpp"
#include <vector>
#include <array>
#include <string>
namespace po167{
template<const int char_size,int base>
struct Trie_Tree
{
struct Node{
std::array<int, char_size> next_node;
std::vector<int> terminate_node;
int parent_node;
int char_number;
int accepet_count;
explicit Node(int c_):parent_node(-1),char_number(c_),accepet_count(0){
for(int i=0;i<char_size;i++) next_node[i]=-1;
}
};
std::vector<Node> nodes;
std::vector<int> fail;
Trie_Tree(){
nodes.push_back(Node(-1));
}
std::vector<int> insert(std::string &word,int word_id){
int node_id=0;
std::vector<int> ids;
for (char & i : word){
node_id=min_insert(i,node_id,word_id);
ids.push_back(node_id);
}
nodes[ids.back()].terminate_node.push_back(word_id);
return ids;
}
int min_insert(char ch,int node_id,int word_id){
int c=(int)(ch-base);
int next_id=nodes[node_id].next_node[c];
if(next_id==-1){
next_id=(int)nodes.size();
nodes.push_back(Node(c));
}
nodes[next_id].parent_node=node_id;
nodes[node_id].next_node[c]=next_id;
// nodes[next_id].accept_node.push_back(word_id);
nodes[node_id].accepet_count++;
node_id=next_id;
return node_id;
}
void aho(){
fail.resize(nodes.size());
std::vector<int> order = {0};
order.reserve(nodes.size());
for (int i = 0; i < (int)(order.size()); i++){
int id = order[i];
if (id == 0){
fail[id] = 0;
}
else{
int f = fail[nodes[id].parent_node];
int c = nodes[id].char_number;
while (f != fail[f] && nodes[f].next_node[c] == -1){
f = fail[f];
}
if (nodes[f].next_node[c] != -1 && nodes[f].next_node[c] != id) fail[id] = nodes[f].next_node[c];
else fail[id] = 0;
}
for (int j = 0; j < char_size; j++){
if (nodes[id].next_node[j] != -1) order.push_back(nodes[id].next_node[j]);
}
}
}
};
}
using po167::Trie_Tree;
#line 4 "test/string/aho.test.cpp"
#include <iostream>
int main() {
std::ios::sync_with_stdio(false);
std::cin.tie(nullptr);
int N;
std::cin >> N;
po167::Trie_Tree<'a', 26> T;
std::vector<int> ans(N);
for (int i = 0; i < N; i++){
std::string S;
std::cin >> S;
auto lis = T.insert(S, i);
ans[i] = lis.back();
}
T.aho();
std::cout << T.nodes.size() << "\n";
for (int i = 1; i < (int)T.nodes.size(); i++){
std::cout << T.nodes[i].parent_node << " " << T.fail[i] << "\n";
}
for (int i = 0; i < N; i++){
std::cout << ans[i] << (i == N - 1 ? "\n" : " ");
}
}