poj3415_Common Substrings
题意
给定两个字符串,求长度大于等于k的公共子串数。
分析
- 将两个字符串中间加个特殊字符拼接,跑后缀数组。
- 将题目转化为对每一个后缀求\(\sum_{j=1}^{i-1}lcp(i,j)\),且后缀\(i\)和\(j\)属于不同字符串。
- 由于\(lcp\)只跟\(h\)数组的区间最小值有关,因此对于单调递减的\(h[i]\)我们可以合并贡献和个数,维护一个单调栈。
- 分别统计\(a\)串对\(b\)的贡献和\(b\)串对\(a\)的贡献。
代码
#include <cstdio>
#include <algorithm>
#include <cstring>
using namespace std;
typedef long long ll;
const int N=3e5+50;
char a[N],b[N],s[N];
int sa[N],rk[N],h[N];
int t[N],t2[N],c[N];
int al,bl,n,k;
void build(int n,int m){
n++;
int *x=t,*y=t2;
for(int i=0;i<m;i++){
c[i]=0;
}
for(int i=0;i<n;i++){
c[x[i]=s[i]]++;
}
for(int i=1;i<m;i++){
c[i]+=c[i-1];
}
for(int i=n-1;i>=0;i--){
sa[--c[x[i]]]=i;
}
for(int k=1;k<=n;k*=2){
int p=0;
for(int i=n-k;i<n;i++){
y[p++]=i;
}
for(int i=0;i<n;i++){
if(sa[i]>=k){
y[p++]=sa[i]-k;
}
}
for(int i=0;i<m;i++){
c[i]=0;
}
for(int i=0;i<n;i++){
c[x[y[i]]]++;
}
for(int i=1;i<m;i++){
c[i]+=c[i-1];
}
for(int i=n-1;i>=0;i--){
sa[--c[x[y[i]]]]=y[i];
}
swap(x,y);
p=1;
x[sa[0]]=0;
for(int i=1;i<n;i++){
x[sa[i]]=y[sa[i-1]]==y[sa[i]] && y[sa[i-1]+k]==y[sa[i]+k]?p-1:p++;
}
if(p>=n){
break;
}
m=p;
}
n--;
for(int i=0;i<=n;i++){
rk[sa[i]]=i;
}
int k=0;
for(int i=0;i<n;i++){
if(k){
k--;
}
int j=sa[rk[i]-1];
while(s[i+k]==s[j+k]){
k++;
}
h[rk[i]]=k;
}
}
void debug(){
for(int i=1;i<=n;i++){
for(int j=sa[i];j<n;j++){
printf("%c",s[j]);
}
printf("\n");
}
for(int i=1;i<=n;i++){
printf("%d ",sa[i]);
}
printf("\n");
for(int i=0;i<n;i++){
printf("%d ",rk[i]);
}
printf("\n");
for(int i=1;i<=n;i++){
printf("%d ",h[i]);
}
printf("\n");
}
//答案就是对任意两个不同后缀a[i...]和b[j...]的sum(lcp(ai,bj)-k+1)
//两个单调栈,一个维护h[i],一个维护贡献之和
ll he[N],ct[N];
ll solve(){
//考虑用单调栈优化到O(n),即对于每一个后缀求与前面后缀的lcp之和,不重不漏
ll ans=0;
//当前后缀与前面每个后缀的lcp之和
//由性质可知,当前后缀和前面某一个后缀的lcp应该是之间的h[i]最小值
//因此可以将递减的h[i]合并为最小的那个h[min]*cnt
ll sum=0;
int tp=0;
for(int i=2;i<=n;i++){
if(h[i]<k){
tp=0;
sum=0;
continue;
}
ll cnt=0;
//维护单调栈,由于lcp只跟区间h最小值有关,将所有栈顶大于当前h[i]的都合并
while(tp && he[tp]>h[i]){
//减去无效栈顶的贡献(h[i]-k+1)
sum-=(he[tp]-k+1)*ct[tp];
//暂时累计cnt,存储到新的栈顶
cnt+=ct[tp];
//栈顶出栈
tp--;
}
//入栈,保持单调性
he[++tp]=h[i];
if(sa[i-1]<al) {
//有效贡献的串,个数加1
cnt++;
}
ct[tp]=cnt;
//累加栈顶贡献
sum+=(he[tp]-k+1)*ct[tp];
if(sa[i]>al){
//将当前累加的贡献加到答案中,即b串后缀与前面所有a串后缀的lcp之和
ans+=sum;
}
}
tp=sum=0;
for(int i=2;i<=n;i++){
if(h[i]<k){
tp=0;
sum=0;
continue;
}
ll cnt=0;
while(tp && he[tp]>h[i]){
sum-=(he[tp]-k+1)*ct[tp];
cnt+=ct[tp];
tp--;
}
if(sa[i-1]>al){
he[++tp]=h[i];
ct[tp]=cnt+1;
sum+=(he[tp]-k+1)*ct[tp];
}else{
he[++tp]=h[i];
ct[tp]=cnt;
sum+=(he[tp]-k+1)*ct[tp];
}
//累加b串后缀与前面所有a串后缀的lcp之和
if(sa[i]<al){
ans+=sum;
}
}
return ans;
}
int main(){
// freopen("in.txt","r",stdin);
while(~scanf("%d",&k) && k){
scanf("%s",a);
scanf("%s",b);
al=strlen(a);
bl=strlen(b);
for(int i=0;i<al;i++){
s[i]=a[i];
}
s[al]='~';
for(int i=0;i<bl;i++){
s[al+1+i]=b[i];
}
n=al+bl+1;
s[n]='\0';
build(n,300);
// debug();
ll ans=solve();
printf("%lld\n",ans);
}
return 0;
}
poj3415_Common Substrings的更多相关文章
- [LeetCode] Unique Substrings in Wraparound String 封装字符串中的独特子字符串
Consider the string s to be the infinite wraparound string of "abcdefghijklmnopqrstuvwxyz" ...
- Leetcode: Unique Substrings in Wraparound String
Consider the string s to be the infinite wraparound string of "abcdefghijklmnopqrstuvwxyz" ...
- CSU-1632 Repeated Substrings (后缀数组)
Description String analysis often arises in applications from biology and chemistry, such as the stu ...
- CF451D Count Good Substrings (DP)
Codeforces Round #258 (Div. 2) Count Good Substrings D. Count Good Substrings time limit per test 2 ...
- LA4671 K-neighbor substrings(FFT + 字符串Hash)
题目 Source http://acm.hust.edu.cn/vjudge/problem/19225 Description The Hamming distance between two s ...
- 后缀数组---New Distinct Substrings
Description Given a string, we need to find the total number of its distinct substrings. Input T- nu ...
- Codeforces Round #258 D Count Good Substrings --计数
题意:由a和b构成的字符串,如果压缩后变成回文串就是Good字符串.问一个字符串有几个长度为偶数和奇数的Good字串. 分析:可知,因为只有a,b两个字母,所以压缩后肯定为..ababab..这种形式 ...
- SPOJ 694. Distinct Substrings (后缀数组不相同的子串的个数)转
694. Distinct Substrings Problem code: DISUBSTR Given a string, we need to find the total number o ...
- Codeforces Round #306 (Div. 2) A. Two Substrings 水题
A. Two Substrings Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://codeforces.com/contest/550/pro ...
随机推荐
- 个人永久性免费-Excel催化剂功能第102波-批量上传本地图片至网络图床(外网可访问)
自我突破,在100+功能后,再做有质量的功能,非常不易,相对录制视频这些轻松活,还是按捺不住去写代码,此功能虽小,但功课也做了不少,希望对真正有需要的群体带来一些惊喜. 背景介绍 图床的使用,一般是写 ...
- 【干货干货】hyperledger fabric 之动态添加组织/修改配置 (Fabric-java-sdk) 下
我们接着上一节来讲: 在熟悉动态增加组织或修改配置的步骤后,我们就可以使用java的api来完成动态增加组织或修改配置了: 废话不多说,直接上干货: 1,预制条件 org3的证书以及组织3的MSP详情 ...
- java:选择排序法对数组排序
最近想练一练Java的算法,然后碰到LeetCode上一道从排序数组删除重复项的小题,刚开始没看到是从排序数组中,就乱写,其实要是排序树组,就比乱序的感觉上好写多了.然后就想回顾下冒泡法对数组排序,凭 ...
- C#3.0新增功能09 LINQ 标准查询运算符 04 运算
连载目录 [已更新最新开发文章,点击查看详细] 本篇主要介绍标准查询运算符的常用运算功能. 01 对数据排序 排序操作基于一个或多个属性对序列的元素进行排序. 第一个排序条件对元素执行主要排序. ...
- webpack基础知识
一.基础 1 安装 npm i -g webpack webpack-cli // 推荐安装至本地 npm i -D webpack webpack-cli 2 webpck基础使用 2.1 webp ...
- linux初学者-延迟及定时任务篇
linux初学者-延迟及定时任务篇 在linux系统的学习工作中,南面会遇到需要延迟进行的任务和需要定时去完成的任务,就像手机的闹钟一样,这时候就需要用到linux系统当中的系统延迟和定时任务的设置了 ...
- Android Studio电脑不支持HAXM的解决办法
Intel HAXM is required to run this AVD. Your CPU does not support required features (VT-x or SVM). U ...
- ListView 控件总结
1.ListView类 1.常用的基本属性: (1)FullRowSelect:设置是否行选择模式.(默认为false) 提示:只有在Details视图该属性才有 ...
- C语言编程入门之--第三章编写第一个C语言程序
第三章 编写第一个C语言程序 导读:一般学一门计算机语言的第一堂上机课(“上机”顾名思义,上了计算机),就是往屏幕输出“hello world”,本章也不例外. 1.1 Hello,World! 这一 ...
- CSS3☞transform变换
transform CSStransform属性允许你旋转,缩放,倾斜或平移给定元素.这是通过修改CSS视觉格式化模型的坐标空间来实现的. DEMO /* Keyword values */ tran ...