题目描述

在神秘的东方有一棵奇葩的树,它有一个固定的根节点(编号为1)。树的每条边上都是一个字符,字符为a,b,c中的一个,你可以从树上的任意一个点出发,然后沿着远离根的边往下行走,在任意一个节点停止,将你经过的边的字符依次写下来,就能得到一个字符串,例如:

在这棵树中我们能够得到的字符串是:

c, cb, ca, a, b, a

现在pty得到了一棵树和一个字符串S。如果S的一个子串[l,r]和树上某条路径所得到的字符串完全相同,则我们称这个子串和该路径匹配。现在pty想知道,S的所有子串和树上的所有路径的匹配总数是多少?

输入格式

第一行:n

接下来n-1行,每行一个整数fa, 一个字符c,字符和整数之间用一个空格隔开

第i行fa代表第i号节点的父亲,c表示第i号节点和fa的连边的字符

最后一行为字符串S

输出格式

输出共一行,表示匹配总数,因为评测系统暂未确定,所以C/C++选手请使用cout输出。

样例

样例输入

5
1 c
2 b
1 a
2 a
cba

样例输出

5

数据范围与提示

【样例说明】

单个字符匹配的对数为4对,两个字符匹配的对数为1对:cb


solution

 
广义后缀自动机
其实是在trie树上建。
具体看
记录 val[i] 表示i这个状态所代表的所有串的出现次数和
val=(max-min)*right
记sum为i到根的val之和。
做匹配的时候,我们考虑新加入一个字符。
假设这个字符的匹配点是i,之前匹配长度为len。
那么这个字符的答案就是(len-s[fa].max)*ri[i]+s[fa].sum.
也就是该字符串在这个点的匹配长度,加上父亲点的sum,也就是以这个字符结尾的其他后缀的贡献。
 格式化代码
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#define maxn 800005
#define ll long long
using namespace std;
int n,id[maxn],rt=,cnt=,la=,a[maxn];
struct node{
int nex[],par,Max,ri;
ll val,sum;
}s[maxn*];
char ch[maxn*];
ll tax[maxn];
void ins(int fa,int c,int i){ int np=++cnt,p=id[fa];s[np].ri=;
//cout<<i<<' '<<c<<' '<<p<<endl;
s[np].Max=s[la].Max+;id[i]=la=np;//这里还不是很懂为什么是la for(;p&&!s[p].nex[c];p=s[p].par)s[p].nex[c]=np;
if(!p)s[np].par=rt;
else {
int q=s[p].nex[c];
if(s[q].Max==s[p].Max+)s[np].par=q;
else {
int nq=++cnt;
for(int i=;i<;i++)s[nq].nex[i]=s[q].nex[i];
s[nq].par=s[q].par;s[nq].Max=s[p].Max+;
s[q].par=s[np].par=nq;
for(;s[p].nex[c]==q;p=s[p].par)s[p].nex[c]=nq;
}
}
}
void Calc(){
for(int i=;i<=cnt;i++)tax[s[i].Max]++;
for(int i=;i<=cnt;i++)tax[i]+=tax[i-];
for(int i=;i<=cnt;i++)a[tax[s[i].Max]--]=i;
for(int i=cnt;i>=;i--){
int k=a[i],f=s[k].par;
s[f].ri+=s[k].ri;
s[k].val=s[k].ri*(s[k].Max-s[f].Max);
}
for(int i=;i<=cnt;i++){
int k=a[i];
s[k].sum=s[k].val+s[s[k].par].sum;
}
}
int main(){
cin>>n;id[]=;
for(int i=,fa;i<=n;i++){
char c;scanf("%d %c",&fa,&c);
ins(fa,c-'a',i);
}
Calc();
scanf("%s",ch);int l=strlen(ch),len=;
int p=rt;ll ans=;
for(int i=;i<l;i++){
for(;!s[p].nex[ch[i]-'a']&&p;p=s[p].par,len=s[p].Max);
if(!p)p=rt,len=;
else {
p=s[p].nex[ch[i]-'a'];len++;
int f=s[p].par;
ans=ans+s[f].sum+(len-s[f].Max)*s[p].ri;
}
}
cout<<ans<<endl;
return ;
}

Pty的字符串(string)的更多相关文章

  1. javascript类型系统——字符串String类型

    × 目录 [1]定义 [2]引号 [3]反斜线[4]特点[5]转字符串 前面的话 javascript没有表示单个字符的字符型,只有字符串String类型,字符型相当于仅包含一个字符的字符串 字符串S ...

  2. C# 字符串string类型转换成DateTime类型 或者 string转换成DateTime?(字符串转换成可空日期类型)

    在c#中,string类型转换成DateTime类型是经常用到的,作为基本的知识,这里在此做个小结.一般来说可以使用多种方法进行转换,最常用的就是使用Convert.ToDateTime(string ...

  3. Java常量字符串String理解

    Java常量字符串String理解 以前关于String的理解仅限于三点:1.String 是final类,不可继承2.String 类比较字符串相等时时不能用“ == ”,只能用  "eq ...

  4. 字符串string类型转换成DateTime或DateTime?类型

    常用的Convert.ToDateTime方法 //将含有正确日期格式的string类型转换成DateTime类型 string strDate = "2014-08-01"; D ...

  5. Javascript基础系列之(三)数据类型 (字符串 String)

    javascript一共有9种数据类型 字符串 String 数值型 Number 布尔型 Boolean 未定义 Undefine 空值 Null 对象 Object 引用Refernce 列表型 ...

  6. Java基础——数组应用之字符串String类

    字符串String的使用 Java字符串就是Unicode字符序列,例如串“Java”就是4个Unicode字符J,a,v,a组成的. Java中没有内置的字符串类型,而是在标准Java类库中提供了一 ...

  7. java中字符串String 转 int(转)

    java中字符串String 转 int String -> int s="12345"; int i; 第一种方法:i=Integer.parseInt(s); 第二种方法 ...

  8. 字符串String类

    1. String类是一个密封类.用关键字sealed修饰: 2. 字符串的两个特性:     ·不可变性:string类型变量,一旦声明就表明它是不会被改变的.因此,string中的方法对strin ...

  9. 字符串string和内存流MemoryStream及比特数组byte[]互转

    原文:字符串string和内存流MemoryStream及比特数组byte[]互转   字符串string和内存流MemoryStream及比特数组byte[]互转比较 定义string变量为str, ...

随机推荐

  1. Java中如何输入一个字符

    今天在QQ群上看见有人问如何在Java中输入一个字符的问题. 查了下有以下三种方法吧: char c = new java.util.Scanner(System.in).next().charAt( ...

  2. ethereum(以太坊)(十三)--异常处理/元祖

    pragma solidity ^0.4.4; contract Students{ uint[] data= new uint[](4); address _owner = msg.sender; ...

  3. PHP时间日期操作增减(date strtotime) 加一天 加一月

    date_default_timezone_set('PRC'); //默认时区 //当前的时间增加5天 $date1 = "2014-11-11"; echo date('Y-m ...

  4. 1014-32-首页13-cell的结构分析---导航栏中间title位置的按钮的尺寸设置---setFrame----

    #import "HWTitleButton.h" #define HWMargin 5 @implementation HWTitleButton - (id)initWithF ...

  5. 部分和问题 南阳acm1058(递归+dfs)

    部分和问题 时间限制:1000 ms  |  内存限制:65535 KB 难度:2   描述 给定整数a1.a2........an,判断是否可以从中选出若干数,使它们的和恰好为K.   输入 首先, ...

  6. python基础之面向对象编程介绍、类和对象

    面向对象变成介绍 面向过程编程 核心是过程(流水线式思维),过程即解决问题的步骤,面向过程的设计就好比精心设计好一条流水线,考虑周全什么时候处理什么东西.主要应用在一旦完成很少修改的地方,如linux ...

  7. js调用js的方法

    a.js文件调用b.js文件时,需要在a.js的第一行添加代码 document.write(" <script language=\"javascript\" s ...

  8. TouTiao开源项目 分析笔记19 问答内容

    1.真实页面预览 1.1.成果预览 首先是问答列表 然后每个item设置点击事件,进入问答内容列表 然后每一个问答内容也设置点击事件,进入问答详情 1.2.触发事件. 在WendaArticleOne ...

  9. iOS程序执行顺序和UIViewController 的生命周期(整理)

    说明:此文是自己的总结笔记,主要参考: iOS程序的启动执行顺序 AppDelegate 及 UIViewController 的生命周期 UIView的生命周期 言叶之庭.jpeg 一. iOS程序 ...

  10. 关于 Inno Setup 报木马的问题处理

    用 Inno Setup 生成的安装包总是报木马,尝试了N次之后发现,把 Compression=lzma 改为 Compression=zip 就不会再报了,可能lzma的压缩算法导致delphi的 ...