洛谷4248 AHOI2013差异 (后缀数组SA+单调栈)

题目链接

补博客!

首先我们观察题目中给的那个求 a n s ans ans的方法,其实前两项没什么用处,直接 f o r for for一遍就求得了

for (int i=1;i<=n;i++) ans=ans+i*(n-1);

那么我们考虑剩下的部分应该怎么求解!
首先这里有一个性质。对于任意两个后缀 i , j i,j i,j,他们的 l c p lcp lcp长度是他们对应的 r a n k rank rank之间的 h e i g h t height height m i n min min (左开右闭)

或者这样说
l c p ( i , j ) = m i n ( h e i g h t [ r a n k [ i ] + 1 ] , h e i g h t [ r a n k [ i ] + 2 ] . . . . . , h e i g h t [ r a n k [ j ] ] ) 其 中 r a n k [ i ] &lt; r a n k [ j ] lcp(i,j) = min(height[rank[i]+1],height[rank[i]+2].....,height[rank[j]]) 其中rank[i]&lt;rank[j] lcp(i,j)=min(height[rank[i]+1],height[rank[i]+2].....,height[rank[j]])rank[i]<rank[j]

那么对于这个题,我们就可以直接维护出每个 h e i g h t height height作为最小值的区间,然后用他的区间个乘上贡献即可(但是具体这里求的时候需要仔细想想,因为那个左开右闭的区间,假设右边能选的端点是 r [ i ] − l + 1 r[i]-l+1 r[i]l+1,那么合法的右端点实际上是由 i − l [ i ] + 1 i-l[i]+1 il[i]+1因为,能覆盖到 l [ i ] l[i] l[i]这个 h e i g h t height height的点实际上是 l [ i ] − 1 l[i]-1 l[i]1。)

总之就是比较难理解啊

for (int i=1;i<=n;i++) ans=ans-2ll*(r[i]-i+1)*(i-l[i]+1)*height[i];

那么现在的问题就是应该怎么求 l [ i ] 和 r [ i ] l[i]和r[i] l[i]r[i]呢?

QWQ这貌似是单调栈的经典应用?

直接从左到右,从右到左扫两遍即可.
这里有一个很好的防止计算重复的方法

就是我们从左到右扫维护的栈是单调的。然后从右到左不单调(非严格)

或者说,一遍单调,一遍不单调,即可解决重复的问题了!

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
#include<cmath>
#include<queue>
#include<map>
#include<set>
#define mk makr_pair
#define ll long long
#define int long long
using namespace std;
inline int read()
{
  int x=0,f=1;char ch=getchar();
  while (!isdigit(ch)) {if (ch=='-') f=-1;ch=getchar();}
  while (isdigit(ch)) {x=(x<<1)+(x<<3)+ch-'0';ch=getchar();}
  return x*f;
}
const int maxn = 2e6+1e2;
struct Node{
    int val,pos;
};
int wb[maxn],sa[maxn];
Node s[maxn];
int l[maxn],r[maxn];
int rk[maxn],h[maxn],height[maxn];
int tmp[maxn];
int n,m;
char a[maxn];
int ans;
void getsa()
{
    int *x = rk,*y = tmp;
    int s = 128;
    int p = 0;
    for (int i=1;i<=n;i++) x[i]=a[i],y[i]=i;
    for (int i=1;i<=s;i++) wb[i]=0;
    for (int i=1;i<=n;i++) wb[x[y[i]]]++;
    for (int i=1;i<=s;i++) wb[i]+=wb[i-1];
    for (int i=n;i>=1;i--) sa[wb[x[y[i]]]--] = y[i];
    for (int j=1;p<n;j<<=1)
    {
        p=0;
        for (int i=n-j+1;i<=n;i++) y[++p]=i;
        for (int i=1;i<=n;i++) if (sa[i]>j) y[++p]=sa[i]-j;
        for (int i=1;i<=s;i++) wb[i]=0;
        for (int i=1;i<=n;i++) wb[x[y[i]]]++;
        for (int i=1;i<=s;i++) wb[i]+=wb[i-1];
        for (int i=n;i>=1;i--) sa[wb[x[y[i]]]--] =y[i];
        swap(x,y);
        p=1;
        x[sa[1]]=1;
        for (int i=2;i<=n;i++)
        {
            x[sa[i]] = (y[sa[i-1]]==y[sa[i]] && y[sa[i]+j]==y[sa[i-1]+j]) ? p : ++p;
        }
        s=p;
    }
    for (int i=1;i<=n;i++) rk[sa[i]]=i;
    h[0]=0;
    for (int i=1;i<=n;i++)
    {
        h[i]=max(h[i-1]-1,(long long)0);
        while (i+h[i]<=n && sa[rk[i]-1]+h[i]<=n && a[i+h[i]]==a[sa[rk[i]-1]+h[i]]) h[i]++;
    }
    for (int i=1;i<=n;i++) height[i] = h[sa[i]];
}
int top;
signed main()
{
  scanf("%s",a+1);
  n = strlen(a+1);
  getsa();
  for (int i=1;i<=n;i++) ans=ans+i*(n-1);
  l[1]=1;
  s[++top].val=height[1];
  s[1].pos=1;
  for (int i=2;i<=n;i++)
  {
  	while (top>=1 && s[top].val>=height[i]) top--;
  	if (!top) l[i]=1;
  	else l[i]=s[top].pos+1;
  	s[++top].val=height[i];
  	s[top].pos=i;
  }	
  memset(s,0,sizeof(s));
  top=1;
  r[n]=n;
  s[top].val=height[n];
  s[top].pos=n;
  for (int i=n-1;i>=1;i--)
  {
  	while (top>=1 && s[top].val>height[i]) top--;
  	if (!top) r[i]=n;
  	else r[i]=s[top].pos-1;
  	s[++top].val=height[i];
  	s[top].pos=i;
  }
  for (int i=1;i<=n;i++) ans=ans-2ll*(r[i]-i+1)*(i-l[i]+1)*height[i];
  cout<<ans;
  return 0;
}

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值