Palindrome

简要题意: 

我们有一个字符串S,字符串的长度不超过500000。

求满足S[i]=S[2n−i]=S[2n+i−2](1≤i≤n)(n≥2)的子串个数。 

分析:

我们能通过简单的数学知识,得到:

该子串是两个回文串拼在一起的,例如abcbabc中,前5项为一个回文串,后5项有一个回文串。

第n项以及第2*n-1项为回文串的中心。

我们可以用Manacher 求得以每个$i$为中心点的回文串半径len[i]。

求得len[i]后,写一个方程:

令i<j,
1. i+len[i]>=j;
2. j-len[j]<=i;
可以推得:
1. i+1<=j<=i+len[i];
2. j-len[j]<=i

所以,这道题便转化成:

已知一个len[i]数组,求满足上列条件的个数! 

怎么求一个区间内小于一个数x的个数呢?

主席树! 

主席树中记录的是每个数字出现过的个数,区间[1,5]即记录数字[1,5]出现的总次数,

我们主席树类似前缀和的思想,依次往后合并。

最终我们求得的答案为:∑query(root[i+len[i]],x)-query(root[i],x);

#include<bits/stdc++.h>
using namespace std;
#define re register int
#define LL long long
const int N=1000005;
int seg,root[N];
struct segment{int v,ls,rs;}tr[N<<4];
int build(int x,int y)
{
int p=++seg;tr[p].v=0;
if(x<y)
{
int mid=(x+y)>>1;
tr[p].ls=build(x,mid);
tr[p].rs=build(mid+1,y);
}
return p;
}
int update(int pre,int l,int r,int x)
{
int p=++seg;
tr[p]=(segment){tr[pre].v+1,tr[pre].ls,tr[pre].rs};
if(l<r)
{
int mid=(l+r)>>1;
if(x<=mid)tr[p].ls=update(tr[pre].ls,l,mid,x);
else tr[p].rs=update(tr[pre].rs,mid+1,r,x);
}
return p;
}
int query(int p1,int p2,int l,int r,int x)
{
if(x<l)return 0;
if(r<=x)return tr[p1].v-tr[p2].v;
int mid=(l+r)>>1;
int res=query(tr[p1].ls,tr[p2].ls,l,mid,x);
if(mid+1<=x)res+=query(tr[p1].rs,tr[p2].rs,mid+1,r,x);
return res;
}
char S[N],s[N<<1];
int val[N<<1],pos[N];
inline void work()
{
scanf("%s",S);int n=strlen(S);
int k=0,id=0,mx=0;
for(re i=0;i<n;++i)s[++k]='#',s[++k]=S[i];s[0]='!';s[++k]='#';
for(re i=1;i<=k;++i)
{
if(i<mx)val[i]=min(val[id*2-i],mx-i);
else val[i]=1;
while(s[i-val[i]]==s[i+val[i]])val[i]++;
if(i+val[i]>mx)mx=i+val[i],id=i;
}
int MAX=0;
for(re i=2;i<=k;i+=2)
{
int id=(i>>1);
pos[id]=(val[i]>>1)-1;
MAX=max(MAX,id-pos[id]);
} seg=0;root[0]=build(1,MAX);
for(re i=1;i<=n;++i)root[i]=update(root[i-1],1,MAX,i-pos[i]);
LL ans=0;
for(re i=1;i<=n;++i)
if(pos[i])ans+=1ll*query(root[i+pos[i]],root[i],1,MAX,i);
printf("%lld\n",ans);
}
signed main()
{
int T;scanf("%d",&T);
while(T--)work();
}

“向前跑,迎着冷眼和嘲笑”

“失败后郁郁寡欢,那是懦夫的表现”----《追梦赤子心》

热身训练3 Palindrome的更多相关文章

  1. 数位dp & 热身训练7

    数位dp 数位dp是一种计数用的dp,一般就是要统计一段区间$[L,R]$内,满足一定条件的数的个数,或者各个数位的个数. 数位dp使得暴力枚举变为满足一定状态的记忆化,更加优秀. 数位dp常常会考虑 ...

  2. 热身训练4 Article

    Article 在这个学期即将结束时,DRD开始写他的最后一篇文章. DRD使用著名的Macrohard的软件World来写他的文章. 不幸的是,这个软件相当不稳定,它总是崩溃. DRD需要在他的文章 ...

  3. 热身训练4 Eighty seven

    Eighty seven 简要题意: n个卡片,其中第i个卡片的数值为$a[i]$.一共q次询问,每次询问将删去其中3个卡片(可能删除若干相同的卡片)后,问能否选出10个卡片,数值之和等于87. n≤ ...

  4. 热身训练2 The All-purpose Zero

    The All-purpose Zero 简要题意:  长度为n的数组,每个数字为S[i],$0$是一种很神奇的数字,你想要的,它都可以变! 问这个序列的最长上升子序列长度为多少? 分析: 我们将除了 ...

  5. 热身训练2 GCD

    题目描述 简要题意:  n个数字,a1,a2,...,an m次询问(l,r),每次询问需回答 1.gcd(al,al+1,al+2,...,ar);2.gcd(ax,ax+1,ax+2,...,ay ...

  6. 热身训练2 Another Meaning

    题目来源 简要题意: 众所周知,在许多情况下,一个词语有两种意思.比如"hehe",不仅意味着"hehe",还意味着"excuse me". ...

  7. 热身训练1 Calculator

    题目出处:Calculator 简要题意: 你有一个确定的函数,f(x)=+...*...^...,其中共有n个操作,从左到右依次计算. 共有m次询问,我们每次询问,1.会修改f(x)中的操作:2.输 ...

  8. 热身训练1 ping ping ping

    点此进入 题意: 一棵树,n+1 个节点,以0号节点为根,给出端点(a,b),节点a到节点b的路径上,至少有一个点是"坏掉的",求"坏掉的点"最少 分析: St ...

  9. 热身训练1 Problem B. Harvest of Apples

    http://acm.hdu.edu.cn/showproblem.php?pid=6333 题意: 求 C(0,n)+C(1,n)+...+C(m,n) 分析: 这道题,我们令s(m,n) = C( ...

随机推荐

  1. HearthbuddyHelper已经开源

    https://github.com/V-arc/HearthbuddyHelper 只是为了稳定,没耗时多久写的一个中控,只是在逐渐堆加,并未进行重构. 通过写这个感受到自己对异步和wpf的理解还有 ...

  2. Linux下运行bash脚本显示“: /usr/bin/env: "bash\r": 没有那个文件或目录

    用 ./ 运行bash脚本文件出现 报错信息 /usr/bin/env: "bash\r": 没有那个文件或目录 错误原因:这主要是因为bash后面多了\r这个字符的原因.在lin ...

  3. 升级到windows10之后的骚操作,安装debian,centos7,支持linux、docker、kubectl命令

    修改Windows10默认字体和图标很大 打开Hyper-V Windows10下载Docker Desktop https://www.docker.com/products/docker-desk ...

  4. 性能再提升70%?大咖前瞻带你揭开.NET6的神秘面纱!

    本月初微软官宣.NET 6 的RC1即将在11月正式发布,这意味着.NET6正式版跟我们见面的时间又近了一步.在之前的.NET6预览版本中,微软加入了大量新功能特性,而在最终版本中将不再额外加入新的内 ...

  5. php move_uploaded_file保存文件失败

    move_uploaded_file保存失败后找错,先使用了try catch,但是没输出信息,才知道该函数在php中是警告属于error,不属于exeption,因此不能通过简单的if(!...)处 ...

  6. Java面向对象系列(1)- 什么是面向对象

    面向过程 & 面向对象 面向过程思想 步骤清晰清楚,第一步做什么,第二步做什么-- 面对过程适合处理一些较为简单的问题 面向对象思想 物以类聚,分类的思维模式,思考问题首先会解决问题需要哪些分 ...

  7. javascript 享元模式 flyweight

    * 适应条件 ** 一个程序中使用了大量的相似对象 造成大的内存开销 ** 对象的大多数状态都可以变为外部状态 ** 剥离出对象的外部状态之后, 可以使用相对较少的共享对象取代大量对象 * 上传文件的 ...

  8. redis小结 1-2

    1.Redis 发布订阅 Redis 发布订阅(pub/sub)是一种消息通信模式:发送者(pub)发送消息,订阅者(sub)接收消息. Redis 客户端可以订阅任意数量的频道. 下图展示了频道 c ...

  9. [转载]session多服务器共享的方案梳理

    转载网址: http://www.cnblogs.com/wangtao_20/archive/2013/10/29/3395518.html session的存储了解以前是怎么做的,搞清楚了来龙去脉 ...

  10. 鸿蒙内核源码分析(原子操作篇) | 是谁在为原子操作保驾护航 | 百篇博客分析OpenHarmony源码 | v34.02

    百篇博客系列篇.本篇为: v34.xx 鸿蒙内核源码分析(原子操作篇) | 谁在为原子操作保驾护航 | 51.c.h .o 本篇说清楚原子操作 读本篇之前建议先读鸿蒙内核源码分析(总目录)系列篇. 基 ...