Description

给出n个结点的树结构T,其中每一个结点上有一个字符,这里我们所说的字符只考虑大写字母A到Z,再给出长度为m
的模式串s,其中每一位仍然是A到z的大写字母。Alice希望知道,有多少对结点<u,v>满足T上从u到V的最短路径
形成的字符串可以由模式串S重复若干次得到?这里结点对<u,v>是有序的,也就是说<u,v>和<v,u>需要被区分.
所谓模式串的重复,是将若干个模式串S依次相接(不能重叠).例如当S=PLUS的时候,重复两次会得到PLUSPLUS,
重复三次会得到PLUSPLUSPLUS,同时要注恿,重复必须是整数次的。例如当S=XYXY时,因为必须重复整数次,所以X
YXYXY不能看作是S重复若干次得到的。

Input

每一个数据有多组测试,
第一行输入一个整数C,表示总的测试个数。
对于每一组测试来说:
第一行输入两个整数,分别表示树T的结点个数n与模式长度m。结点被依次编号为1到n,
之后一行,依次给出了n个大写字母(以一个长度为n的字符串的形式给出),依次对应树上每一个结点上的字符(
第i个字符对应了第i个结点).
之后n-1行,每行有两个整数u和v表示树上的一条无向边,之后一行给定一个长度为m的由大写字母组成的字符串,
为模式串S。
1<=C<=10,3<=N<=10000003<=M<=1000000

Output

给出C行,对应C组测试。每一行输出一个整数,表示有多少对节点<u,v>满足从u到v的路径形成的字符串恰好是模
式串的若干次重复.

点分治统计答案,hash判定当前点到中心的路径是否是由模式串重复得到的串的前/后缀,为保证复杂度,递归至子树大小不足m时结束,时间复杂度O(Tnlog(n/m))

#include<bits/stdc++.h>
int _(){
int x=,c=getchar();
while(c<)c=getchar();
while(c>)x=x*+c-,c=getchar();
return x;
}
typedef unsigned long long u64;
const int N=1e6+,p=;
int T,n,m;
char s1[N],s2[N];
int es[N*],enx[N*],e0[N],ep=,sz[N],SZ,CG;
u64 h1[N],h2[N],ans;
bool ed[N];
void f2(int w,int pa){
sz[w]=;
for(int i=e0[w],u;i;i=enx[i]){
u=es[i];
if(u==pa||ed[u])continue;
f2(u,w);
sz[w]+=sz[u];
}
}
int md,dw;
void f3(int w,int pa){
if(++dw>md)md=dw;
bool is=sz[w]*>=SZ;
for(int i=e0[w],u;i;i=enx[i]){
u=es[i];
if(u==pa||ed[u])continue;
f3(u,w);
if(sz[u]*>SZ)is=;
}
if(is)CG=w;
--dw;
}
int mod[N];
int t1[N],t2[N],a1[N],a2[N];
void f4(int w,int pa,u64 h,int dep){
h=h*p+s1[w];
if(h==h1[dep])++a1[dep[mod]];
if(h==h2[dep])++a2[dep[mod]];
for(int i=e0[w],u;i;i=enx[i]){
u=es[i];
if(u==pa||ed[u])continue;
f4(u,w,h,dep+);
}
}
void f1(int w){
f2(w,);
SZ=sz[w];
md=dw=;
f3(w,);
w=CG;
ed[w]=;
if(md*<m||SZ<m)return;
memset(t1,,sizeof(int)*(m+));
memset(t2,,sizeof(int)*(m+));
for(int i=e0[w];i;i=enx[i]){
int u=es[i];
if(ed[u])continue;
memset(a1,,sizeof(int)*(m+));
memset(a2,,sizeof(int)*(m+));
f4(u,w,s1[w],);
a1[m]=a1[];
a2[m]=a2[];
a1[m+]=a1[];
a2[m+]=a2[];
ans+=a1[]+a2[];
for(int x=;x<m;++x){
ans+=u64(a1[m+-x])*t2[x];
ans+=u64(a2[m+-x])*t1[x];
}
for(int x=;x<m;++x)t1[x]+=a1[x],t2[x]+=a2[x];
}
for(int i=e0[w];i;i=enx[i]){
int u=es[i];
if(!ed[u])f1(u);
}
}
int main(){
for(T=_();T;--T){
ans=;
n=_();m=_();
memset(e0,,sizeof(int)*(n+));
memset(ed,,n+);
scanf("%s",s1+);
for(int i=,a,b;i<n;++i){
a=_();b=_();
es[ep]=b;enx[ep]=e0[a];e0[a]=ep++;
es[ep]=a;enx[ep]=e0[b];e0[b]=ep++;
}
scanf("%s",s2+);
u64 pp=;
for(int i=,j=,k=m;i<=n;++i,pp*=p){
mod[i]=i%m;
h1[i]=s2[j]*pp+h1[i-];
h2[i]=s2[k]*pp+h2[i-];
if(m<++j)j=;
if(!--k)k=m;
}
if(m>)f1();
else for(int i=;i<=n;++i)if(s1[i]==s2[])++ans;
printf("%llu\n",ans);
}
return ;
}

bzoj4598: [Sdoi2016]模式字符串的更多相关文章

  1. BZOJ4598 [Sdoi2016]模式字符串 【点分治 + hash】

    题目 给出n个结点的树结构T,其中每一个结点上有一个字符,这里我们所说的字符只考虑大写字母A到Z,再给出长度为m 的模式串s,其中每一位仍然是A到z的大写字母.Alice希望知道,有多少对结点< ...

  2. Bzoj4598: [Sdoi2016]模式字符串 点分治 哈希

    国际惯例的题面:这种关于树上路径的题,我也没什么好办法,只好点分治.考虑当前分治重心为root,如何统计经过分治重心的路径的答案.我们令prf[i]表示某个点到root的路径(不含root)已经循环匹 ...

  3. BZOJ4598: [Sdoi2016]模式字符串(点分治 hash)

    题意 题目链接 Sol 直接考虑点分治+hash匹配 设\(up[i]\)表示\(dep \% M = i\)的从下往上恰好与前\(i\)位匹配的个数 \(down\)表示\(dep \% M = i ...

  4. 【BZOJ4598】[Sdoi2016]模式字符串 树分治+hash

    [BZOJ4598][Sdoi2016]模式字符串 Description 给出n个结点的树结构T,其中每一个结点上有一个字符,这里我们所说的字符只考虑大写字母A到Z,再给出长度为m的模式串s,其中每 ...

  5. P4075 [SDOI2016]模式字符串

    总结 P4075 [SDOI2016]模式字符串 题目描述 给出n个结点的树结构T,其中每一个结点上有一个字符,这里我们所说的字符只考虑大写字母A到Z,再给出长度为m的模式串s,其中每一位仍然是A到z ...

  6. [SDOI2016] 模式字符串 (BZOJ4598 & VIJOS1995)

    首先直接点分+hash就可以做,每个点用hash判断是否为S重复若干次后的前缀或后缀,每个子树与之前的结果O(m)暴力合并.在子树大小<m时停止分治,则总复杂度为O(nlog(n/m)). 问题 ...

  7. bzoj 4598: [Sdoi2016]模式字符串

    题目描述 给出n个结点的树结构T,其中每一个结点上有一个字符,这里我们所说的字符只考虑大写字母A到Z,再给出长度为m的模式串s,其中每一位仍然是A到z的大写字母. Alice希望知道,有多少对结点&l ...

  8. [SDOI2016]模式字符串

    Description 给出n个结点的树结构T,其中每一个结点上有一个字符,这里我们所说的字符只考虑大写字母A到Z,再给出长度为m的模式串s,其中每一位仍然是A到z的大写字母.Alice希望知道,有多 ...

  9. BZOJ.4598.[SDOI2016]模式字符串(点分治 Hash)

    LOJ BZOJ 洛谷 点分治.考虑如何计算过\(rt\)的答案. 记\(pre[i]\)表示(之前的)子树内循环匹配了\(S\)的前缀\(i\)的路径有多少,\(suf[i]\)表示(之前的)子树内 ...

随机推荐

  1. bzoj2501

    题解: 显然,每当进入一个小的边界,那么我们的ans+1,出去一个大的边界,ans-1 然后,我们将每一个边界排序,时间小的在前,大的在后 每一次进来一个,如果是左边的边界,+1,右边的-1 然后输出 ...

  2. 【HTML5】使用 JavaScript 来获取电池状态(Battery Status API)

    HTML5 规范已经越来越成熟,可以让你访问更多来自设备的信息,其中包括最近提交的 "Battery Status API".如其名称所示,该 API 允许你通过 JavaScri ...

  3. php获取excel文件数据

    很简单就可以实现,下面为大家简单介绍下 1.下载PHPExcel类,是一个文件夹,还得有一个文件PHPExcel.php,两个在同级目录 require __DIR__ . './PHPExcel/I ...

  4. Kotlin都转正成Android官方语言了,你还不试一下?

    想想Android Studio 和Eclipse ,我觉得你还是有必要入手Kotlin了. 站好队很重要. 以前的一篇总体概括老文,大家可以看看~ 爽翻天!告别Java.一起来使用kotlin开发完 ...

  5. linux uname和dpkg命令

    uname -a:查看系统一些参数 dpkg -i:安装下载好的.deb包裹

  6. 程序设计入门-C语言基础知识-翁恺-第一周:简单的计算程序-详细笔记(一)

    目录 第一周:简单的计算程序 1.1 第一个程序 Hello World! 1.2 变量 1.3 计算 1.4 编程作业及课后讨论 第一周:简单的计算程序 1.1 第一个程序 Hello World! ...

  7. HDU2181 哈密顿绕行世界问题

    解题思路:哈密顿环游世界问题.一道简单的题目,用回溯. #include<cstdio> #include<cstring> #include<algorithm> ...

  8. HDU3394Railway Tarjan连通算法

    There are some locations in a park, and some of them are connected by roads. The park manger needs t ...

  9. redhat7学习笔记之从零到部署javaweb项目

    REDHAT7学习笔记 1. 安装vmware10 安装过程略,下载地址:链接: https://pan.baidu.com/s/16odKKkRYBxGWDVo1cz_wxA 注意,10以上版本不在 ...

  10. BZOJ1690 Usaco2007 Dec 奶牛的旅行 【01分数规划】

    BZOJ1690 Usaco2007 Dec 奶牛的旅行 题目描述 作为对奶牛们辛勤工作的回报,Farmer John决定带她们去附近的大城市玩一天.旅行的前夜,奶牛们在兴奋地讨论如何最好地享受这难得 ...