关闭

回文树总结

标签: 回文树回文串
391人阅读 评论(0) 收藏 举报
分类:

最近学习了回文树,这个比较新颖的数据结构,相应的写了12道关于回文树的题目。所以总结一下。

网络上关于回文树的学习的博客有很多质量很好的,这里就不具体分析回文树的构成了,但是我想把自己对回文树的理解写一下。


首先回文树是两个树,每个节点都是一个回文串。先说节点吧,节点是一个回文子串,但是不记录整个回文子串,而是记录回文串的长度,和两个指针

next指针,和fail指针。这是最基础的,当然我们可以在节点上再增加别的信息。

 next指针一个二维数组,next [ i ] [ j ] ,指向的是第i个节点在两端加上 j 对应的字符形成的回文串节点。next指针也是回文树主要支架,回文树是两棵树。每棵树节点之间都是next指针连接起来,这个和字典树相似。那么fail指针,是一个数组,指向的是该节点回文串中的最长后缀回文串的节点,fail指针式连接两棵树的边,相当于两棵树之间有很多边相互连接,同一棵树上也是有fail将两个节点连接起来的。

下图实线就是next指针,虚线是fail指针




一直说回文树是两棵树,两棵树的区别是,一棵树的节点是长度为偶数的回文串,另一棵树是长度为奇数的回文串。一开始会建立长度是0的节点和长度是-1的节点。长度为0就表示是偶回文串的树的根节点,而长度为-1就表示奇回文树的根节。


这是字符串aaba形成的回文树,aaba中包含四种回文串,a,b,aba,是长度为奇数的回文串,aa,是长度为偶数的回文串,a是aa的最长后缀回文子串,也是aba的最长后缀回文子串。实线是形成两棵树的基础,而虚线则是穿插在两棵树之间的桥梁


下面给出回文树的模板,再具体分析:

struct Tree
{
    int next[MAX+5][26];//next指针
    int num[MAX+5];//当前节点表示回文串中的后缀子回文串的数目
    int cnt[MAX+5];//当前节点表示回文串的个数
    int fail[MAX+5];//fail指针
    int len[MAX+5];//当前节点表示回文串的长度
    int s[MAX+5];//储存字符串
    int p;//节点个数
    int last;//最后一个节点
    int n;//字符串长度
    int new_node(int x)
    {
        memset(next[p],0,sizeof(next[p]));
        cnt[p]=0;
        num[p]=0;
        len[p]=x;
        return p++;
    }
    void init()
    {
        p=0;
        new_node(0);
        new_node(-1);
        last=0;
        n=0;
        s[0]=-1;
        fail[0]=1;
    }
    int get_fail(int x)
    {
        while(s[n-len[x]-1]!=s[n])
            x=fail[x];
        return x;
    }
    int add(int x)
    {
        x-='a';
        s[++n]=x;
        int cur=get_fail(last);
        if(!(last=next[cur][x]))
        {
            int now=new_node(len[cur]+2);
            fail[now]=next[get_fail(fail[cur])][x];
            next[cur][x]=now;
            num[now]=num[fail[now]]+1;
            last=now;
            return 1;
        }
        cnt[last]++;
        return 0;
    }
    void count()
    {
        for(int i=p-1;i>=0;p++)
            cnt[fail[i]]+=cnt[i];
    }
}tree;

如何构造回文树就不赘述了。这里我们分析这种网络上大量普及的模板可以做到哪些功能?

1,字符串中本质不同的回文串的种数。

这里p就是节点个数,也代表不同回文串的种数,每个节点代表的回文串都是独一无二的。

2,字符串中每个本质不同的回文串的个数   即cnt数组。

3,字符串中以某个点为结束或者开始形成的回文串的个数。

4,字符串中回文串总数。

第三个功能,某个点为结束,是通过num数组实现的,每次插入一个字符的时候,要么形成新的回文串要么形成旧的回文串,总之,在插入的时候返回num[last]就可以了。如果以某个点为开始,把字符串倒着输入就可以了

第四个功能就是第一个和第二个结合起来,当然也可以通过第三个稍加计算也可以得出来。

这些是基本的功能,了解这些就可以做一些比较基础的回文串的题目。但是作为ACMer,我们不能止步于这么简单的内容!


下面给出一些比较难的情况:

1,字符串不是从左往右依次给你的,它是一个可以在两头加的不断延长的字符,可以在前面加,也可有在后面加。这个时候应该怎么处理?

首先s数组应该是可以从两边加的,所以s数组设成字符串长度的2倍,那么一开始我们是在中间的,由于要在两边加入字符串,所以我们应该设置两个last指针

表示左边最后一个节点,和右边最后一个节点。那么也要设置两个边界,字符串的边界,表示从中间,左边的字符串长度,右边的字符串长度。这样的话就可以在两边加上字符串。但是你会发现一个问题,当在左边加入一个字符的时候,有可能连着右边的最后一个字符形成一个整的的回文串,那么对于右边而言last就要变了,就要变成整的回文串,所以要进行特判!


2,如果可以删除字符串怎么办?字符串可以加入,也可以删除字符串,当然从结尾开始删

如果加入字符形成了一个新的节点,那么要把该节点删除,即p- -。同时这个字符也形成了指向自己的next指针要把next指针删掉。

如果没有形成新的节点,只需要n- - 就好了,


3,如果现在回文树不是对一个字符串操作,而是要你对两个字符串操作,应该怎么办?

建立两棵回文树?,但是如果要你比较两个字符串有多少公共的回文串子串,两棵树的节点一一比较吗?显然超时。所以在一棵树上进行操作。在插入第一个字符串的时候,我们不用担心,可是插入第二个字符串,肯定会被前面的字符干扰啊。这个时候我们有两种解决办法:

1,可以在两个字符串之间插入2个两个字符串都没有出现过的字符,这样插入第二个字符串的时候肯定不会和第一个字符串形成回文串。

2,插入完第二个字符串的时候,把

        last=0;
        s[0]=-1;
        fail[0]=1;
        n=0;

建立新的数组num,cnt 表示第二个字符串的相关信息。




    


     


0
0
查看评论

回文树练习

回文树的题目一般只有几种类型,主要是应用cnt,num 1.ural1960. Palindromes and Super Abilities http://acm.timus.ru/problem.aspx?space=1&num=1960 题意:每加入一个字符,询问不同回文串种...
  • alpc_wt
  • alpc_wt
  • 2015-09-16 13:02
  • 815

回文树(Palindromic Tree)+黑科技 学习笔记

回文树(Palindromic Tree)最基本的回文树在网上用很多资料,在这里做简单的介绍。本文的重点是后面回文树的一些更广泛的应用。注:本文的图片转自http://adilet.org/blog/25-09-14/,所以图中某些变量的定义可能与本文不同,需要注意。定义节点回文树中的每个节点都对应...
  • YxuanwKeith
  • YxuanwKeith
  • 2017-03-24 22:48
  • 752

回文树/回文自动机 Palindromic Tree 学习小记

前言这个东西呢,是由战斗民族的信息选手Mikhail Rubinchik搞出来的一个数据结构,正如其名,就是用来解决回文相关的题目的。应该说,是manacher的一个特殊化,所以他跟manacher有很多相似之处。整体感知这是由两棵树组成的东西,一棵树存长度为奇数的回文串-tr1,另一个存偶数的-t...
  • ZLTJohn
  • ZLTJohn
  • 2016-02-09 23:46
  • 899

回文树介绍(Palindromic Tree)

简介回文树是由Mikhail Rubinchik大神发明的,在Petrozavodsk Summer Camp 2014上首次提出来,是一个很新的数据结构,目前相关资料比较少。顾名思义,回文树是一个用来解决回文串相关问题。回文树的结构就像线段树、平衡树等其它树结构一样,回文树由若干个节点组成,每个节...
  • lwfcgz
  • lwfcgz
  • 2015-09-26 00:12
  • 4801

回文树(Palindrome Tree)/回文自动机(Palindrome Automaton)学习小记

简介回文树(回文自动机),是解决一类回文串问题的强大数据结构,比ManacherManacher扩展了很多功能。 这个数据结构比较新,由来自战斗民族的神犇MikhailRubinchik在2014年的Petrozavodsk夏令营提出。 这个数据结构代码量其实超级少。必备技能ManacherMa...
  • a_crazy_czy
  • a_crazy_czy
  • 2016-01-31 17:51
  • 1419

Palindromic Tree 回文自动机-回文树 解决回文串的神器

回文树,也叫回文自动机,是2014年夏天战斗民族发明的,其功能如下: 1、求前缀字符串中的本质不同的回文串种类 2、求每个本质不同回文串的个数 3、以下标i为结尾的回文串个数/种类 4、每个本质不同回文串包含的本质不同回文串种类 (本文参考自Palindromic Tree——回文树【处理...
  • alpc_wt
  • alpc_wt
  • 2015-09-16 12:13
  • 1937

Palindromic Tree——回文树【处理一类回文串问题的强力工具】

今天我们来学习一个神奇的数据结构:Palindromic Tree。中译过来就是——回文树。 那么这个回文树有何功能? 假设我们有一个串S,S下标从0开始,则回文树能做到如下几点: 1.求串S前缀0~i内本质不同回文串的个数(两个串长度不同或者长度相同且至少有一个字符不同便是本质不同) 2.求串S...
  • u013368721
  • u013368721
  • 2014-12-23 16:14
  • 14461

回文树笔记(转自quack_quack)

1.回文树的next[charset]指针: b->aba 那么就这样表示:b.next[a]=aba 当然树里面肯定不能存字符串,于是就直接用下标标号代替了 2.回文树的fail指针: 跟ac自动机类似,fail指针指向当前节点的最大回文后缀 没有就指向根 3.回...
  • geng4512
  • geng4512
  • 2015-12-24 22:44
  • 905

BZOJ3676:[Apio2014]回文串 (Manacher+后缀自动机+树上倍增)

题目传送门:http://www.lydsy.com/JudgeOnline/problem.php?id=3676题目分析:找后缀自动机练习题的时候看到这题的,结果想不出怎么用SAM。看了黄学长的博客才知道要先写个manacher,再在parent树上玩倍增。后来又听说这题是裸的回文树……我还不知...
  • KsCla
  • KsCla
  • 2017-07-07 14:01
  • 196

占有欲

<br />占有欲出现的年龄<br />  在一岁前或三岁后的孩子们身上,占有欲表现不会很强烈,而在一到三岁之间孩子的占有欲表现最强。这和儿童的活动方式、自我意识发展的水平有密切关系。<br />   1岁前的孩子,基本上是个体活动。他和其他孩子交往时,也会发生...
  • shelley_huhu
  • shelley_huhu
  • 2011-03-02 15:44
  • 300
    个人资料
    • 访问:130994次
    • 积分:4657
    • 等级:
    • 排名:第7453名
    • 原创:349篇
    • 转载:0篇
    • 译文:0篇
    • 评论:7条
    最新评论