补博客!
首先我们观察题目中给的那个求\(ans\)的方法,其实前两项没什么用处,直接\(for\)一遍就求得了
for (int i=1;i<=n;i++) ans=ans+i*(n-1);
那么我们考虑剩下的部分应该怎么求解!
首先这里有一个性质。对于任意两个后缀
\(i,j\),他们的
\(lcp\)长度是他们对应的
\(rank\)之间的
\(height\)的
\(min\) (左开右闭)
或者这样说
\(lcp(i,j) = min(height[rank[i]+1],height[rank[i]+2].....,height[rank[j]]) 其中rank[i]<rank[j]\) 那么对于这个题,我们就可以直接维护出每个\(height\)作为最小值的区间,然后用他的区间个乘上贡献即可(但是具体这里求的时候需要仔细想想,因为那个左开右闭的区间,假设右边能选的端点是\(r[i]-l+1\),那么合法的右端点实际上是由\(i-l[i]+1\)因为,能覆盖到\(l[i]\)这个\(height\)的点实际上是\(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]\)呢?
QWQ这貌似是单调栈的经典应用?
直接从左到右,从右到左扫两遍即可.
这里有一个很好的防止计算重复的方法
就是我们从左到右扫维护的栈是单调的。然后从右到左不单调(非严格)
或者说,一遍单调,一遍不单调,即可解决重复的问题了!
#include #include #include #include #include #include #include