最近学习了一下关于Trie的一些姿势,感觉很实用。

终于不用每次看到字符串判重等操作就只想到hash

关于Trie的定义,来自百度百科

在计算机科学中,Trie,又称前缀树或字典树,是一种有序树状的数据结构,用于保存关联数组,其中的键通常是字符串。

说的有点高级,我们不要管它,可以看这样一张图:

这棵Trie中的单词共有:his,he,her,me,your

就是从根节点一直到某个有end(结尾)标记的点

可以发现,Tire其实就是把一些字符串的公用前缀字符存储了下来

Tire的实现更是简单:

  1. 建立一棵Trie,初始时只有一个空的根节点(编号为0)

  2. 每当插入或删除时,如果当前字符在之前的操作中已经建立,然后直接利用即可,否则新建立一个节点并让上一个节点连上它

  3. 当一个字符串结束时,给该节点做个记号,同时也可存储些其它的信息

具体的实现方法也有两种:邻接表邻接矩阵

在Trie中,显然用邻接矩阵是很快的,但空间的开销可能较大,如果题目说明范围,那么空间允许的情况下可以使用

邻接表还是一样,比较省空间(毕竟要多少开多少)

这里用一道板子题来理解一下:

hihocoder 1366 : 逆序单词

邻接矩阵CODE

#include<iostream>
#include<string>
using namespace std;
const int N=50005;
struct node
{
bool end;
int next[30];
}trie[N<<4];
string s;
int n,ans,cnt;
inline bool find(string s)
{
int now=0,len=s.size();
for (register int i=0;i<len;++i)
{
if (trie[now].next[s[i]-'a'+1]) now=trie[now].next[s[i]-'a'+1]; else return 0;
if (i==len-1) return trie[now].end;
}
}
inline void insert(string s)
{
int now=0,len=s.size();
for (register int i=0;i<len;++i)
{
if (trie[now].next[s[i]-'a'+1]) now=trie[now].next[s[i]-'a'+1]; else trie[now].next[s[i]-'a'+1]=++cnt,now=cnt;
if (i==len-1) trie[now].end=1;
}
}
int main()
{
register int i;
for (cin>>n,i=1;i<=n;++i)
{
cin>>s;
string rs(s.rbegin(),s.rend());
if (find(rs)) ++ans;
insert(s);
}
cout<<ans;
return 0;
}

邻接表CODE

#include<iostream>
#include<cstring>
#include<string>
using namespace std;
const int N=50005;
struct node
{
bool end;
char ch;
}trie[N<<4];
struct edge
{
int to,next;
}link[N<<4];
string s;
int head[N<<4],n,ans,cnt;
inline void add(int x,int y,char z)
{
link[y].to=y; trie[y].ch=z; link[y].next=head[x]; head[x]=y;
}
inline bool find(string s)
{
int now=0,len=s.size();
for (register int i=0;i<len;++i)
{
bool flag=0;
for (register int j=head[now];j!=-1;j=link[j].next)
if (trie[link[j].to].ch==s[i]) { flag=1; now=link[j].to; break; }
if (!flag) return 0;
if (i==len-1) return trie[now].end;
}
}
inline void insert(string s)
{
int now=0,len=s.size();
for (register int i=0;i<len;++i)
{
bool flag=0;
for (register int j=head[now];j!=-1;j=link[j].next)
if (trie[link[j].to].ch==s[i]) { flag=1; now=link[j].to; break; }
if (!flag) add(now,++cnt,s[i]),now=cnt;
if (i==len-1) trie[now].end=1;
}
}
int main()
{
register int i;
memset(link,-1,sizeof(link));
memset(head,-1,sizeof(head));
for (cin>>n,i=1;i<=n;++i)
{
cin>>s;
string rs(s.rbegin(),s.rend());
if (find(rs)) ++ans;
insert(s);
}
cout<<ans;
return 0;
}

关于Trie的一些算法的更多相关文章

  1. [算法]从Trie树(字典树)谈到后缀树

    我是好文章的搬运工,原文来自博客园,博主July_,地址:http://www.cnblogs.com/v-July-v/archive/2011/10/22/2316412.html 从Trie树( ...

  2. Trie树的创建、插入、查询的实现

    原文:http://blog.chinaunix.net/xmlrpc.php?r=blog/article&uid=28977986&id=3807947 1.什么是Trie树 Tr ...

  3. Aho-Corasick算法、多模正则匹配、Snort入门学习

    希望解决的问题 . 在一些高流量.高IO的WAF中,是如何对规则库(POST.GET)中的字符串进行多正则匹配的,是单条轮询执行,还是多模式并发执行 . Snort是怎么组织.匹配高达上千条的正则规则 ...

  4. [转载]字典树(trie树)、后缀树

    (1)字典树(Trie树) Trie是个简单但实用的数据结构,通常用于实现字典查询.我们做即时响应用户输入的AJAX搜索框时,就是Trie开始.本质上,Trie是一颗存储多个字符串的树.相邻节点间的边 ...

  5. trie树(前缀树)

    问题描述:   Trie树,即字典树,又称单词查找树或键树,是一种树形结构,是一种哈希树的变种.典型应用是用于统计和排序大量的字符串(但不仅限于字符串),所以经常被搜索引擎系统用于文本词频统计.它的优 ...

  6. Trie树(字典树) 最热门的前N个搜索关键词

    方法介绍 1.1.什么是Trie树 Trie树,即字典树,又称单词查找树或键树,是一种树形结构.典型应用是用于统计和排序大量的字符串(但不仅限于字符串),所以经常被搜索引擎系统用于文本词频统计.它的优 ...

  7. 双数组trie树的基本构造及简单优化

    一 基本构造 Trie树是搜索树的一种,来自英文单词"Retrieval"的简写,可以建立有效的数据检索组织结构,是中文匹配分词算法中词典的一种常见实现.它本质上是一个确定的有限状 ...

  8. 从Trie树(字典树)谈到后缀树

    转:http://blog.csdn.net/v_july_v/article/details/6897097 引言 常关注本blog的读者朋友想必看过此篇文章:从B树.B+树.B*树谈到R 树,这次 ...

  9. Trie树 - 字典树

    1.1.什么是Trie树 Trie树,即字典树,又称单词查找树或键树,是一种树形结构.典型应用是用于统计和排序大量的字符串(但不仅限于字符串),所以经常被搜索引擎系统用于文本词频统计.它的优点是最大限 ...

随机推荐

  1. android 解决连接电视机顶盒失败的方法

    今天在开发过程中,需要连接海美迪的电视盒子,这个盒子是基于android6.0的版本,之前连接其它电视盒子都正常,当输入 adb -s xxxx shell后,盒子连接失败,日志如下: error: ...

  2. LeetCode题解之 Increasing Order Search Tree

    1.题目描述 2/问题分析 利用中序遍历,然后重新构造树. 3.代码 TreeNode* increasingBST(TreeNode* root) { if (root == NULL) retur ...

  3. windows7下搭建python环境并用pip安装networkx

    1.安装顺序:Python+pip+pywin32+numpy+matplotlib+networkx 2.版本问题 所安装的所有程序和包都需要具有统一的python版本.系统版本和位宽,所以第一步要 ...

  4. Unix操作系统监控详解(一)

    一.描述 监控在检查系统问题运行状况以及优化系统性能工作上是一个不可缺少的部分.通过操作系统监控工具监视操作系统资源的使用情况,间接地反映了各服务器程序的运行情况.根据运行结果分析可以帮助我们快速定位 ...

  5. .net core 在linux系统运行

    .net都已经跨平台了,所以想把一些东西部署到linux服务器上去 ,首先介绍一款叫做MobaXterm的软件,功能相当强大,感觉比xshell和putty好用,可以相对方便的操作linux系统,官网 ...

  6. sql server 2008 身份验证失败 18456

    双击打开后加上  ;-m  然后以管理员方式  打开 SQLSERVER 2008  就可以已window身份登录  不过还没有完 右键  属性  =>安全性 更改为 sql server 和 ...

  7. CheckTimeWait.bat实现windows下的TimeWait检查

    原文链接: http://www.lookdaima.com/WebForms/WebPages/Blanks/Pm/Docs/DocItemDetail.aspx?id=c7bff196-cd9c- ...

  8. BZOJ5092:[Lydsy1711月赛]分割序列(贪心,高维前缀和)

    Description 对于一个长度为n的非负整数序列b_1,b_2,...,b_n,定义这个序列的能量为:f(b)=max{i=0,1,...,n}((b_1 xor b_2 xor...xor b ...

  9. 【洛谷】【单调栈】P1901 发射站

    [题目描述:] 某地有 N 个能量发射站排成一行,每个发射站 i 都有不相同的高度 Hi,并能向两边(当 然两端的只能向一边)同时发射能量值为 Vi 的能量,并且发出的能量只被两边最近的且比 它高的发 ...

  10. android 登录效果

    <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android=&q ...